diff --git a/packages/sdk-codegen-scripts/src/reformatter.ts b/packages/sdk-codegen-scripts/src/reformatter.ts index de13b6094..04115e1fe 100644 --- a/packages/sdk-codegen-scripts/src/reformatter.ts +++ b/packages/sdk-codegen-scripts/src/reformatter.ts @@ -295,6 +295,28 @@ class GoFormatter extends BaseFormatter { } } +class ProtoFormatter extends BaseFormatter { + constructor() { + super('Protobuf') + } + + versionStamp() { + return warn('Skipping SDK version updating - not implemented for Protobuf.') + } +} + +class GrpcProxyFormatter extends BaseFormatter { + constructor() { + super('Grpc') + } + + versionStamp() { + return warn( + 'Skipping SDK version updating - not implemented for Grpc proxy.' + ) + } +} + type IFormatFiles = { [key: string]: string[] } type IFormatters = { [key: string]: IReformat } @@ -306,6 +328,8 @@ const fileFormatters: IFormatters = { '.swift': new SwiftFormatter(), '.ts': new TypescriptFormatter(), '.go': new GoFormatter(), + '.proto': new ProtoFormatter(), + '.java': new GrpcProxyFormatter(), } export class FilesFormatter { diff --git a/packages/sdk-codegen/src/codeGenerators.ts b/packages/sdk-codegen/src/codeGenerators.ts index 22099fe4c..e74ab6cab 100644 --- a/packages/sdk-codegen/src/codeGenerators.ts +++ b/packages/sdk-codegen/src/codeGenerators.ts @@ -32,6 +32,8 @@ import { SwiftGen } from './swift.gen' import { PythonGen } from './python.gen' import { TypescriptGen } from './typescript.gen' import { GoGen } from './go.gen' +import { ProtoGen } from './proto.gen' +import { GrpcProxyGen } from './grpc_proxy.gen' export interface IGeneratorSpec { /** source code file extension regex */ @@ -101,6 +103,18 @@ export const Generators: Array = [ options: '-papiPackage=Looker -ppackageName=looker', extension: /\.php/gi, }, + { + factory: (api: ApiModel, versions?: IVersionInfo) => + new ProtoGen(api, versions), + language: 'Protobuf', + extension: /\.proto/gi, + }, + { + factory: (api: ApiModel, versions?: IVersionInfo) => + new GrpcProxyGen(api, versions), + language: 'GrpcProxy', + extension: /\.java/gi, + }, // { // language: 'R', // legacy: 'r' diff --git a/packages/sdk-codegen/src/grpc_proxy.gen.spec.ts b/packages/sdk-codegen/src/grpc_proxy.gen.spec.ts new file mode 100644 index 000000000..3d9b63d73 --- /dev/null +++ b/packages/sdk-codegen/src/grpc_proxy.gen.spec.ts @@ -0,0 +1,52 @@ +/* + + MIT License + + Copyright (c) 2021 Looker Data Sciences, Inc. + + Permission is hereby granted, free of charge, to any person obtaining a copy + of this software and associated documentation files (the "Software"), to deal + in the Software without restriction, including without limitation the rights + to use, copy, modify, merge, publish, distribute, sublicense, and/or sell + copies of the Software, and to permit persons to whom the Software is + furnished to do so, subject to the following conditions: + + The above copyright notice and this permission notice shall be included in all + copies or substantial portions of the Software. + + THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, + FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE + AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER + LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, + OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE + SOFTWARE. + + */ + +import { TestConfig } from './testUtils' +import { GrpcProxyGen } from './grpc_proxy.gen' + +const config = TestConfig() +const apiTestModel = config.apiTestModel + +const gen = new GrpcProxyGen(apiTestModel) + +describe('pseudocode', () => { + describe('method signature', () => { + it('optional body and additional param', () => { + const method = apiTestModel.methods.create_user_credentials_email + expect(method).toBeDefined() + const expected = `create_user_credentials_email(user_id, body, fields): CredentialsEmail` + const actual = gen.methodSignature('', method) + expect(actual).toEqual(expected) + }) + it('no params', () => { + const method = apiTestModel.methods.all_datagroups + expect(method).toBeDefined() + const expected = `all_datagroups(): Datagroup[]` + const actual = gen.methodSignature('', method) + expect(actual).toEqual(expected) + }) + }) +}) diff --git a/packages/sdk-codegen/src/grpc_proxy.gen.ts b/packages/sdk-codegen/src/grpc_proxy.gen.ts new file mode 100644 index 000000000..26b8a6192 --- /dev/null +++ b/packages/sdk-codegen/src/grpc_proxy.gen.ts @@ -0,0 +1,282 @@ +/* + + MIT License + + Copyright (c) 2021 Looker Data Sciences, Inc. + + Permission is hereby granted, free of charge, to any person obtaining a copy + of this software and associated documentation files (the "Software"), to deal + in the Software without restriction, including without limitation the rights + to use, copy, modify, merge, publish, distribute, sublicense, and/or sell + copies of the Software, and to permit persons to whom the Software is + furnished to do so, subject to the following conditions: + + The above copyright notice and this permission notice shall be included in all + copies or substantial portions of the Software. + + THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, + FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE + AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER + LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, + OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE + SOFTWARE. + + */ + +import { CodeGen } from './codeGen' +import type { IMethod, IParameter, IProperty, IType } from './sdkModels' +import { titleCase, camelCase } from './sdkModels' + +// eslint-disable @typescript-eslint/no-unused-vars + +/** + * Pseudocde generator + */ +export class GrpcProxyGen extends CodeGen { + codePath = './proto/grpc_proxy/src/main/java/com/google/looker/server' + packagePath = '' + sdkPath = 'sdk' + itself = '' + fileExtension = '.java' + commentStr = '// ' + nullStr = 'null' + transport = 'transport' + + argDelimiter = ', ' + paramDelimiter = ',\n' + propDelimiter = '\n' + codeQuote = '"' + enumDelimiter = ',\n' + + indentStr = ' ' + endTypeStr = '\n}' + needsRequestTypes = false + willItStream = true + + private readonly defaultApi = '4.0' + + isDefaultApi() { + return this.apiVersion === this.defaultApi + } + + supportsMultiApi() { + return false + } + + sdkFileName(baseFileName: string) { + if (baseFileName === 'streams') { + return this.fileName('sdk/LookerStreamingServiceImpl') + } else if (baseFileName === 'models') { + return this.fileName('sdk/LookerModels') + } else { + return this.fileName('sdk/LookerServiceImpl') + } + } + + /** + * Grpc Proxy Server generator + * + * @param {string} indent indentation for code + * @param {IMethod} method for signature + * @returns {string} prototype declaration of method + */ + methodSignature(indent: string, method: IMethod): string { + indent = '' + const params = method.allParams + const args = params.map((p) => p.name) + return `${indent}${method.operationId}(${args.join(', ')}): ${ + method.primaryResponse.type.name + }` + } + + construct(_indent: string, _type: IType): string { + return '' + } + + declareMethod(_indent: string, _method: IMethod): string { + const titleMethodName = titleCase(_method.operationId) + const camelMethodName = camelCase(_method.operationId) + return `${this.formatJavaDoc(_method.description)} + @Override + public void ${camelMethodName}(${titleMethodName}Req request, StreamObserver<${titleMethodName}Response> responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.${_method.httpMethod.toLowerCase()}("${ + _method.endpoint + }", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ${titleMethodName}Response.Builder responseBuilder = ${titleMethodName}Response.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + ` + } + + declareStreamer(_indent: string, _method: IMethod): string { + const titleMethodName = titleCase(_method.operationId) + const camelMethodName = camelCase(_method.operationId) + const returnCanStream = _method.returnType?.type.name.endsWith('[]') + const streamResponse = _method.returnType?.type.name.endsWith('[]') + ? 'Stream' + : '' + const onNext = returnCanStream + ? `responseBuilder.getResultList().forEach(entry -> { + ${titleMethodName}StreamResponse.Builder responseBuilder2 = ${titleMethodName}StreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + });` + : `responseObserver.onNext(responseBuilder.build());` + return `${this.formatJavaDoc(_method.description)} + @Override + public void ${camelMethodName}(${titleMethodName}Req request, StreamObserver<${titleMethodName}${streamResponse}Response> responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.${_method.httpMethod.toLowerCase()}("${ + _method.endpoint + }", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ${titleMethodName}Response.Builder responseBuilder = ${titleMethodName}Response.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + ${onNext} + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + ` + } + + declareParameter( + _indent: string, + _method: IMethod, + _param: IParameter + ): string { + return '' + } + + declareProperty(_indent: string, _property: IProperty): string { + return '' + } + + encodePathParams(_indent: string, _method: IMethod): string { + return '' + } + + methodsEpilogue(_indent: string): string { + return '}' + } + + methodsPrologue(_indent: string): string { + return this.servicesPrologue('LookerService') + } + + streamsPrologue(_indent: string): string { + return this.servicesPrologue('LookerStreamingService') + } + + modelsEpilogue(_indent: string): string { + return '}' + } + + modelsPrologue(_indent: string): string { + return ` +package com.google.looker.server.sdk; + +// DELETE THIS FILE - NOT REQUIRED + +public class LookerModels { + ` + } + + beginRegion(_: string, description: string): string { + return ` //#region ${description}` + } + + endRegion(_: string, description: string): string { + return ` //#endregion ${description}` + } + + declareType() { + return '' + } + + summary(_indent: string, _text: string | undefined): string { + return '' + } + + typeSignature(_indent: string, _type: IType): string { + return '' + } + + private servicesPrologue(serviceName: string) { + return ` +package com.google.looker.server.sdk; + +import com.google.looker.grpc.services.*; +import com.google.looker.grpc.services.${serviceName}Grpc.${serviceName}ImplBase; +import com.google.looker.server.rtl.LookerClient; +import com.google.looker.server.rtl.LookerClientResponse; +import com.google.protobuf.InvalidProtocolBufferException; +import com.google.protobuf.util.JsonFormat; +import io.grpc.Status; +import io.grpc.stub.StreamObserver; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +public class ${serviceName}Impl extends ${serviceName}ImplBase { + + final private static Logger LOGGER = LoggerFactory.getLogger(${serviceName}Impl.class); + + final private LookerClient lookerClient; + + public ${serviceName}Impl() { + lookerClient = new LookerClient("${this.apiVersion}"); + } + + ` + } + + private formatJavaDoc(comments: string) { + if (comments.trim().length === 0) { + return '' + } else { + const lines = comments.split('\n').map((part) => ` * ${part}\n`) + lines.unshift(' /**\n') + lines.push(' */') + return lines.join('') + } + } +} diff --git a/packages/sdk-codegen/src/proto.gen.spec.ts b/packages/sdk-codegen/src/proto.gen.spec.ts new file mode 100644 index 000000000..fbf681c9e --- /dev/null +++ b/packages/sdk-codegen/src/proto.gen.spec.ts @@ -0,0 +1,52 @@ +/* + + MIT License + + Copyright (c) 2021 Looker Data Sciences, Inc. + + Permission is hereby granted, free of charge, to any person obtaining a copy + of this software and associated documentation files (the "Software"), to deal + in the Software without restriction, including without limitation the rights + to use, copy, modify, merge, publish, distribute, sublicense, and/or sell + copies of the Software, and to permit persons to whom the Software is + furnished to do so, subject to the following conditions: + + The above copyright notice and this permission notice shall be included in all + copies or substantial portions of the Software. + + THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, + FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE + AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER + LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, + OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE + SOFTWARE. + + */ + +import { TestConfig } from './testUtils' +import { ProtoGen } from './proto.gen' + +const config = TestConfig() +const apiTestModel = config.apiTestModel + +const gen = new ProtoGen(apiTestModel) + +describe('ProtoGen', () => { + describe('method signature', () => { + it('optional body and additional param', () => { + const method = apiTestModel.methods.create_user_credentials_email + expect(method).toBeDefined() + const expected = `create_user_credentials_email(user_id, body, fields): CredentialsEmail` + const actual = gen.methodSignature('', method) + expect(actual).toEqual(expected) + }) + it('no params', () => { + const method = apiTestModel.methods.all_datagroups + expect(method).toBeDefined() + const expected = `all_datagroups(): Datagroup[]` + const actual = gen.methodSignature('', method) + expect(actual).toEqual(expected) + }) + }) +}) diff --git a/packages/sdk-codegen/src/proto.gen.ts b/packages/sdk-codegen/src/proto.gen.ts new file mode 100644 index 000000000..a1cfd628d --- /dev/null +++ b/packages/sdk-codegen/src/proto.gen.ts @@ -0,0 +1,331 @@ +/* + + MIT License + + Copyright (c) 2021 Looker Data Sciences, Inc. + + Permission is hereby granted, free of charge, to any person obtaining a copy + of this software and associated documentation files (the "Software"), to deal + in the Software without restriction, including without limitation the rights + to use, copy, modify, merge, publish, distribute, sublicense, and/or sell + copies of the Software, and to permit persons to whom the Software is + furnished to do so, subject to the following conditions: + + The above copyright notice and this permission notice shall be included in all + copies or substantial portions of the Software. + + THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, + FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE + AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER + LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, + OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE + SOFTWARE. + + */ + +import { CodeGen } from './codeGen' +import type { IMethod, IParameter, IProperty, IType } from './sdkModels' +import { EnumType, snakeCase, stringToHashCode, titleCase } from './sdkModels' + +// eslint-disable @typescript-eslint/no-unused-vars + +/** + * Protobuf and grpc generator + */ +export class ProtoGen extends CodeGen { + saveMethods: IMethod[] = [] + codePath = './proto/grpc_proxy/src/main/proto' + packagePath = '' + sdkPath = 'sdk' + itself = '' + fileExtension = '.proto' + commentStr = '// ' + nullStr = 'null' + transport = 'transport' + + argDelimiter = ', ' + paramDelimiter = ',\n' + propDelimiter = '\n' + codeQuote = '"' + enumDelimiter = ',\n' + + indentStr = ' ' + endTypeStr = '\n}' + needsRequestTypes = false + willItStream = true + + private readonly defaultApi = '4.0' + + isDefaultApi() { + return this.apiVersion === this.defaultApi + } + + supportsMultiApi() { + return false + } + + sdkFileName(baseFileName: string) { + return this.fileName(`sdk/${baseFileName}`) + } + + /** + * grpc signature generator + * + * @param {string} indent indentation for code + * @param {IMethod} method for signature + * @returns {string} prototype declaration of method + */ + methodSignature(indent: string, method: IMethod): string { + indent = '' + const params = method.allParams + const args = params.map((p) => p.name) + return `${indent}${method.operationId}(${args.join(', ')}): ${ + method.primaryResponse.type.name + }` + } + + construct(_indent: string, _type: IType): string { + return '' + } + + declareMethod(_indent: string, _method: IMethod): string { + this.saveMethods.push(_method) + const methodName = titleCase(_method.operationId) + return `${this.formatComments( + _method.description + )} rpc ${methodName}(${methodName}Req) returns (${methodName}Response);` + } + + declareStreamer(_indent: string, _method: IMethod): string { + const methodName = titleCase(_method.operationId) + const streamResponse = _method.returnType?.type.name.endsWith('[]') + ? 'Stream' + : '' + return `${this.formatComments( + _method.description + )} rpc ${methodName}(${methodName}Req) returns (stream ${methodName}${streamResponse}Response);` + } + + declareParameter( + _indent: string, + _method: IMethod, + _param: IParameter + ): string { + return '' + } + + declareProperty(_indent: string, _property: IProperty): string { + return `${this.formatComments(_property.description)} ${this.mapType( + _property.type.name + )} ${_property.name} = ${this.generateIdentifier(_property.name)};\n` + } + + encodePathParams(_indent: string, _method: IMethod): string { + return '' + } + + methodsEpilogue(_indent: string): string { + return '}' + } + + methodsPrologue(_indent: string): string { + return this.servicesPrologue('LookerService') + } + + streamsPrologue(_indent: string): string { + return this.servicesPrologue('LookerStreamingService') + } + + modelsEpilogue(_indent: string): string { + return this.saveMethods + .map((method) => { + const isStreamResponse = method.returnType?.type.name.endsWith('[]') + let streamResponse = '' + if (isStreamResponse) { + console.log( + `stream response ${method.name} ${method.returnType?.type.name}` + ) + streamResponse = ` + +message ${titleCase(method.operationId)}StreamResponse { + ${this.methodResponse(method, 1, true).trim()} +}` + } + return ` +message ${titleCase(method.operationId)}Req { +${this.methodArguments(method)} +} + +message ${titleCase(method.operationId)}Response { + ${this.methodResponse(method, 1).trim()} +}${streamResponse} +` + }) + .join('') + } + + modelsPrologue(_indent: string): string { + return ` +syntax = "proto3"; + +package looker; + +option java_package = "com.google.looker.grpc.services"; +option java_multiple_files = true; + +import "google/protobuf/any.proto"; +import "google/protobuf/timestamp.proto"; + ` + } + + summary(_indent: string, _text: string | undefined): string { + return '' + } + + typeSignature(_indent: string, _type: IType): string { + return '' + } + + beginRegion(_: string, description: string): string { + return ` // ${description}` + } + + endRegion(): string { + return '' + } + + declareType(_: string, type: IType) { + let propertyValues = '' + let typeType + if (type instanceof EnumType) { + typeType = 'enum' + const num = type as EnumType + const typeName = snakeCase(type.name).toUpperCase() + const enumValues = num.values.map((enumType) => { + const enumName = `${typeName}_${enumType.toString().toUpperCase()}` + return ` ${enumName} = ${this.generateIdentifier(enumName)};\n` + }) + enumValues.unshift(` _${typeName}_UNSET = 0;\n`) + propertyValues = enumValues.join('') + } else { + typeType = 'message' + propertyValues = Object.values(type.properties) + .map((prop) => { + return this.declareProperty('', prop) + }) + .join('') + } + + return `${this.formatComments(type.description).trim()} +${typeType} ${type.name} { + ${propertyValues.trim()} +}` + } + + private servicesPrologue(serviceName: string) { + return ` +syntax = "proto3"; + +package looker; + +option java_package = "com.google.looker.grpc.services"; +option java_multiple_files = true; + +import 'sdk/models.proto'; + +service ${serviceName} { + ` + } + + private formatComments(comments: string) { + return comments.trim().length === 0 + ? '' + : comments + .split('\n') + .map((part) => ` // ${part}\n`) + .join('') + } + + private methodArguments(method: IMethod) { + return ( + ' ' + + method.allParams + .map((param) => { + return `${this.formatComments(param.description)} ${this.mapType( + param.type.name + )} ${param.name} = ${this.generateIdentifier(param.name)};\n` + }) + .join('') + .trim() + ) + } + + private methodResponse( + method: IMethod, + index: number, + isStreamResponse = false + ) { + if (method.returnType) { + const description = this.formatComments( + method.returnType?.description || '' + ) + const returnType = method.returnType.type.name + if (returnType === 'void') { + return `${description}` + } else { + return `${description} ${this.mapType( + returnType, + isStreamResponse + )} result = ${index};\n` + } + } else { + return '' + } + } + + private mapType(type: string, isStreamResponse = false): string { + if (type.startsWith('Hash[')) { + return `map` + } else if (type.endsWith('[]')) { + if (isStreamResponse) { + return `${type.substring(0, type.length - 2)}` + } else { + return `repeated ${type.substring(0, type.length - 2)}` + } + } else if (type === 'boolean') { + return 'bool' + } else if (type === 'datetime') { + return 'google.protobuf.Timestamp' + } else if (type === 'any' || type === 'object') { + return 'google.protobuf.Any' + } else if (type === 'uri') { + return 'string' + } else if (type.startsWith('DelimArray')) { + // TODO handle this better + return 'string' + } else { + return type + } + } + + // Not convinced about this implementation but will do + // for now. Originally used the index of property in + // javascript object but this is a little brittle as + // there is no guarantee a developer will not insert + // a new property into the object. This generates a + // consistent value across runs. The problem is that the + // value MUST be between 0 and 536870911. To fix this + // negative values are multipled by -1. Values greater + // than 536870911 are bitwise shift right until they are + // less than equal to 536870911. So far their have been + // no collisions but I suspect there are better implementations. + private generateIdentifier(name: string): number { + let hashCode = stringToHashCode(name.split('').reverse().join('')) + hashCode = hashCode < 0 ? hashCode * -1 : hashCode + while (hashCode > 536870911) { + hashCode = hashCode >> 1 + } + return hashCode + } +} diff --git a/packages/sdk-codegen/src/sdkModels.ts b/packages/sdk-codegen/src/sdkModels.ts index 1bbb3c7c3..165337feb 100644 --- a/packages/sdk-codegen/src/sdkModels.ts +++ b/packages/sdk-codegen/src/sdkModels.ts @@ -139,6 +139,16 @@ export const titleCase = (value: string) => { return value[0].toLocaleUpperCase() + value.substr(1) } +/** + * convert string to snake_case + * @param value string value to convert to snake_case + */ +export const snakeCase = (value: string) => { + if (!value) return '' + value = value.charAt(0).toLowerCase() + value.substring(1) + return value.replace(/[A-Z]/g, (letter) => `_${letter.toLowerCase()}`) +} + /** * Only first character of string should be uppercase * @@ -153,6 +163,18 @@ export const firstCase = (value: string) => { return value[0].toLocaleUpperCase() + value.substr(1).toLocaleLowerCase() } +/** + * Derive a hash from a string. + * @param s string to hash + */ +export const stringToHashCode = (s: string): number => { + let h = 0 + for (let n = 0; n < s.length; n++) { + h = ((h << 5) - h + s.charCodeAt(n)) | 0 + } + return h +} + export interface IModel {} /** diff --git a/proto/grpc_proxy/.env_sample b/proto/grpc_proxy/.env_sample new file mode 100644 index 000000000..6282f2a48 --- /dev/null +++ b/proto/grpc_proxy/.env_sample @@ -0,0 +1,13 @@ +GRPC_SERVER_HOST=localhost +GRPC_SERVER_LISTEN_PORT=50051 +CERT_CHAIN_FILE=ssl/server.crt +PRIVATE_KEY_FILE=ssl/server.pem +TRUST_MANAGER_FILE="ssl/ca.crt" +LOOKER_CLIENT_ID= +LOOKER_CLIENT_SECRET= +LOOKER_BASE_URL=https://self-signed.looker.com:19999 +LOOKER_VERIFY_SSL=false +TEST_LOOKER_USERNAME= +TEST_LOOKER_PASSWORD= +TEST_CONNECTION_NAME= + diff --git a/proto/grpc_proxy/.gitignore b/proto/grpc_proxy/.gitignore new file mode 100644 index 000000000..0211bd25f --- /dev/null +++ b/proto/grpc_proxy/.gitignore @@ -0,0 +1,7 @@ +.env +.gradle +build +out +ssl +src/main/java/com/google/looker/server/sdk/LookerModels.java +bin/ \ No newline at end of file diff --git a/proto/grpc_proxy/README.md b/proto/grpc_proxy/README.md new file mode 100644 index 000000000..7b5dae9d4 --- /dev/null +++ b/proto/grpc_proxy/README.md @@ -0,0 +1,94 @@ +# Looker GRPC Proxy Server + +GRPC proxy server to a Looker instance. +1. Listens for GRPC requests. +2. Converts GRPC request to Looker JSON. +3. Calls Looker JSON rest endpoint. +4. Converts rest response to GRPC response. +5. Returns GRPC response. + +## Setup + +Install protobuf generator, `protoc`. +`brew install protobuf` + +File formatter. +`brew install clang-format` + +### SSL setup for local testing + +Run `scripts/ssl_setup.sh` + +### Generate protobuf definitions and java implementation (TODO) + +Run `scripts/gen_protobuf.sh` + +### .env setup + +Prior to server startup create a `.env` file in the root of this project (note +that it should not be added to source control). A sample file, `.env_sample`, +contains the entries required. + +``` +# host grpc server listens on. Used by the grpc client. +GRPC_SERVER_HOST=localhost +# port grpc server listens on. Used by grpc server and client. +GRPC_SERVER_LISTEN_PORT=50051 +# Certificate chain file. Used by server to support SSL setup for development. +CERT_CHAIN_FILE=ssl/server.crt +# Private key file. Used by server to support SSL setup for development. +PRIVATE_KEY_FILE=ssl/server.pem +# Trust manager file. Used by client to support SSL setup for development. +TRUST_MANAGER_FILE="ssl/ca.crt" +# Looker client id +LOOKER_CLIENT_ID= +# Looker client secret +LOOKER_CLIENT_SECRET= +# Looker server base url +LOOKER_BASE_URL=https://self-signed.looker.com:19999 +# Verify ssl. Set to false for development environmet +LOOKER_VERIFY_SSL=false +# Looker connection db username - used by ConnectionTests +TEST_LOOKER_USERNAME= +# Looker connection db password - used by ConnectionTests +TEST_LOOKER_PASSWORD= +# Looker connection name - used by ConnectionTests to test a connection +TEST_CONNECTION_NAME= +``` + +## Notes + +### Protobuf identifier generation + +Not convinced about the implementation that generates protobuf identifiers but +it will do for now. Originally it used the index of the property in javascript +object but this is a little brittle as there is no guarantee a developer will +not insert a new property into the object. This generates a consistent value +across runs. The problem is that the identifier MUST be between 0 and 536870911. +To fix this negative values are multipled by -1 and values greater than +536870911 are bitwise shifted right until they are less than or equal to +536870911. So far their have been no collisions but I suspect there are better +implementations. + +## TODOs + +In no particular order of importance. + +1. Streaming support. +2. Add rest endpoint to protobuf files. +3. Sync Looker server environment variable names with other implementations. +4. Handle response content types other that JSON. +5. Industrialize generation of proto ids (handle slight possibility of duplicates). +Verify or improve current id generator. +6. Tests for generators. +7. Tests for java support functions. +8. Add support for productionized SSL connection. +9. Separate client into another project. Tests utilizing should go with the client. +10. Consider creating a new runtime that can be embedded in helltool. Note sure it +can be done. Basically it would call the internal ruby API endpoint and negate the +need for an extra network hop. Gets http2 for free? +11. Finish authentication methods. +12. Implement refresh token. +13. Rewrite setup script in typescript and to codegen scripts package. + + diff --git a/proto/grpc_proxy/build.gradle b/proto/grpc_proxy/build.gradle new file mode 100644 index 000000000..d449b2465 --- /dev/null +++ b/proto/grpc_proxy/build.gradle @@ -0,0 +1,60 @@ +plugins { + id 'java' + id 'com.google.protobuf' version '0.8.14' + id 'idea' +} + +group 'com.google.looker' +version '1.0-SNAPSHOT' + +repositories { + mavenCentral() +} + +protobuf { + protoc { + artifact = "com.google.protobuf:protoc:3.12.0" + } + plugins { + grpc { + artifact = 'io.grpc:protoc-gen-grpc-java:1.34.1' + } + } + generateProtoTasks { + all()*.plugins { + grpc {} + } + } +} + +dependencies { + implementation 'javax.annotation:javax.annotation-api:1.3.2' + implementation 'io.grpc:grpc-netty-shaded:1.34.1' + implementation 'io.grpc:grpc-protobuf:1.34.1' + implementation 'io.grpc:grpc-stub:1.34.1' + implementation 'org.apache.commons:commons-lang3:3.11' + implementation 'com.google.protobuf:protobuf-java-util:3.14.0' + implementation 'io.github.cdimascio:java-dotenv:5.2.2' + implementation "io.ktor:ktor-client:$ktorVersion" + implementation "io.ktor:ktor-client-okhttp:$ktorVersion" + implementation "io.ktor:ktor-client-json:$ktorVersion" + implementation "io.ktor:ktor-client-jackson:$ktorVersion" + implementation 'org.slf4j:slf4j-api:1.7.30' + implementation 'org.slf4j:slf4j-simple:1.7.30' + + testImplementation 'org.junit.jupiter:junit-jupiter-api:5.6.0' + testRuntimeOnly 'org.junit.jupiter:junit-jupiter-engine' +} + +test { + useJUnitPlatform() +} + +// if you have source imports issues, add the below +sourceSets.main.java.srcDir new File(buildDir, 'generated/source') +idea { + module { + // Marks the already(!) added srcDir as "generated" + generatedSourceDirs += file('build/generated/source') + } +} diff --git a/proto/grpc_proxy/gradle.properties b/proto/grpc_proxy/gradle.properties new file mode 100644 index 000000000..f88853c87 --- /dev/null +++ b/proto/grpc_proxy/gradle.properties @@ -0,0 +1,3 @@ +org.gradle.daemon=true +org.gradle.jvmargs=-Xmx2560m +ktorVersion=1.4.2 diff --git a/proto/grpc_proxy/gradle/wrapper/gradle-wrapper.jar b/proto/grpc_proxy/gradle/wrapper/gradle-wrapper.jar new file mode 100644 index 000000000..f3d88b1c2 Binary files /dev/null and b/proto/grpc_proxy/gradle/wrapper/gradle-wrapper.jar differ diff --git a/proto/grpc_proxy/gradle/wrapper/gradle-wrapper.properties b/proto/grpc_proxy/gradle/wrapper/gradle-wrapper.properties new file mode 100644 index 000000000..ef503b1c3 --- /dev/null +++ b/proto/grpc_proxy/gradle/wrapper/gradle-wrapper.properties @@ -0,0 +1,6 @@ +#Sun Jan 10 17:12:20 PST 2021 +distributionUrl=https\://services.gradle.org/distributions/gradle-6.1.1-all.zip +distributionBase=GRADLE_USER_HOME +distributionPath=wrapper/dists +zipStorePath=wrapper/dists +zipStoreBase=GRADLE_USER_HOME diff --git a/proto/grpc_proxy/gradlew b/proto/grpc_proxy/gradlew new file mode 100755 index 000000000..2fe81a7d9 --- /dev/null +++ b/proto/grpc_proxy/gradlew @@ -0,0 +1,183 @@ +#!/usr/bin/env sh + +# +# Copyright 2015 the original author or authors. +# +# 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. +# + +############################################################################## +## +## Gradle start up script for UN*X +## +############################################################################## + +# Attempt to set APP_HOME +# Resolve links: $0 may be a link +PRG="$0" +# Need this for relative symlinks. +while [ -h "$PRG" ] ; do + ls=`ls -ld "$PRG"` + link=`expr "$ls" : '.*-> \(.*\)$'` + if expr "$link" : '/.*' > /dev/null; then + PRG="$link" + else + PRG=`dirname "$PRG"`"/$link" + fi +done +SAVED="`pwd`" +cd "`dirname \"$PRG\"`/" >/dev/null +APP_HOME="`pwd -P`" +cd "$SAVED" >/dev/null + +APP_NAME="Gradle" +APP_BASE_NAME=`basename "$0"` + +# Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. +DEFAULT_JVM_OPTS='"-Xmx64m" "-Xms64m"' + +# Use the maximum available, or set MAX_FD != -1 to use that value. +MAX_FD="maximum" + +warn () { + echo "$*" +} + +die () { + echo + echo "$*" + echo + exit 1 +} + +# OS specific support (must be 'true' or 'false'). +cygwin=false +msys=false +darwin=false +nonstop=false +case "`uname`" in + CYGWIN* ) + cygwin=true + ;; + Darwin* ) + darwin=true + ;; + MINGW* ) + msys=true + ;; + NONSTOP* ) + nonstop=true + ;; +esac + +CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar + +# Determine the Java command to use to start the JVM. +if [ -n "$JAVA_HOME" ] ; then + if [ -x "$JAVA_HOME/jre/sh/java" ] ; then + # IBM's JDK on AIX uses strange locations for the executables + JAVACMD="$JAVA_HOME/jre/sh/java" + else + JAVACMD="$JAVA_HOME/bin/java" + fi + if [ ! -x "$JAVACMD" ] ; then + die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME + +Please set the JAVA_HOME variable in your environment to match the +location of your Java installation." + fi +else + JAVACMD="java" + which java >/dev/null 2>&1 || die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. + +Please set the JAVA_HOME variable in your environment to match the +location of your Java installation." +fi + +# Increase the maximum file descriptors if we can. +if [ "$cygwin" = "false" -a "$darwin" = "false" -a "$nonstop" = "false" ] ; then + MAX_FD_LIMIT=`ulimit -H -n` + if [ $? -eq 0 ] ; then + if [ "$MAX_FD" = "maximum" -o "$MAX_FD" = "max" ] ; then + MAX_FD="$MAX_FD_LIMIT" + fi + ulimit -n $MAX_FD + if [ $? -ne 0 ] ; then + warn "Could not set maximum file descriptor limit: $MAX_FD" + fi + else + warn "Could not query maximum file descriptor limit: $MAX_FD_LIMIT" + fi +fi + +# For Darwin, add options to specify how the application appears in the dock +if $darwin; then + GRADLE_OPTS="$GRADLE_OPTS \"-Xdock:name=$APP_NAME\" \"-Xdock:icon=$APP_HOME/media/gradle.icns\"" +fi + +# For Cygwin or MSYS, switch paths to Windows format before running java +if [ "$cygwin" = "true" -o "$msys" = "true" ] ; then + APP_HOME=`cygpath --path --mixed "$APP_HOME"` + CLASSPATH=`cygpath --path --mixed "$CLASSPATH"` + JAVACMD=`cygpath --unix "$JAVACMD"` + + # We build the pattern for arguments to be converted via cygpath + ROOTDIRSRAW=`find -L / -maxdepth 1 -mindepth 1 -type d 2>/dev/null` + SEP="" + for dir in $ROOTDIRSRAW ; do + ROOTDIRS="$ROOTDIRS$SEP$dir" + SEP="|" + done + OURCYGPATTERN="(^($ROOTDIRS))" + # Add a user-defined pattern to the cygpath arguments + if [ "$GRADLE_CYGPATTERN" != "" ] ; then + OURCYGPATTERN="$OURCYGPATTERN|($GRADLE_CYGPATTERN)" + fi + # Now convert the arguments - kludge to limit ourselves to /bin/sh + i=0 + for arg in "$@" ; do + CHECK=`echo "$arg"|egrep -c "$OURCYGPATTERN" -` + CHECK2=`echo "$arg"|egrep -c "^-"` ### Determine if an option + + if [ $CHECK -ne 0 ] && [ $CHECK2 -eq 0 ] ; then ### Added a condition + eval `echo args$i`=`cygpath --path --ignore --mixed "$arg"` + else + eval `echo args$i`="\"$arg\"" + fi + i=`expr $i + 1` + done + case $i in + 0) set -- ;; + 1) set -- "$args0" ;; + 2) set -- "$args0" "$args1" ;; + 3) set -- "$args0" "$args1" "$args2" ;; + 4) set -- "$args0" "$args1" "$args2" "$args3" ;; + 5) set -- "$args0" "$args1" "$args2" "$args3" "$args4" ;; + 6) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" ;; + 7) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" ;; + 8) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" ;; + 9) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" "$args8" ;; + esac +fi + +# Escape application args +save () { + for i do printf %s\\n "$i" | sed "s/'/'\\\\''/g;1s/^/'/;\$s/\$/' \\\\/" ; done + echo " " +} +APP_ARGS=`save "$@"` + +# Collect all arguments for the java command, following the shell quoting and substitution rules +eval set -- $DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS "\"-Dorg.gradle.appname=$APP_BASE_NAME\"" -classpath "\"$CLASSPATH\"" org.gradle.wrapper.GradleWrapperMain "$APP_ARGS" + +exec "$JAVACMD" "$@" diff --git a/proto/grpc_proxy/gradlew.bat b/proto/grpc_proxy/gradlew.bat new file mode 100644 index 000000000..24467a141 --- /dev/null +++ b/proto/grpc_proxy/gradlew.bat @@ -0,0 +1,100 @@ +@rem +@rem Copyright 2015 the original author or authors. +@rem +@rem Licensed under the Apache License, Version 2.0 (the "License"); +@rem you may not use this file except in compliance with the License. +@rem You may obtain a copy of the License at +@rem +@rem https://www.apache.org/licenses/LICENSE-2.0 +@rem +@rem Unless required by applicable law or agreed to in writing, software +@rem distributed under the License is distributed on an "AS IS" BASIS, +@rem WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +@rem See the License for the specific language governing permissions and +@rem limitations under the License. +@rem + +@if "%DEBUG%" == "" @echo off +@rem ########################################################################## +@rem +@rem Gradle startup script for Windows +@rem +@rem ########################################################################## + +@rem Set local scope for the variables with windows NT shell +if "%OS%"=="Windows_NT" setlocal + +set DIRNAME=%~dp0 +if "%DIRNAME%" == "" set DIRNAME=. +set APP_BASE_NAME=%~n0 +set APP_HOME=%DIRNAME% + +@rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. +set DEFAULT_JVM_OPTS="-Xmx64m" "-Xms64m" + +@rem Find java.exe +if defined JAVA_HOME goto findJavaFromJavaHome + +set JAVA_EXE=java.exe +%JAVA_EXE% -version >NUL 2>&1 +if "%ERRORLEVEL%" == "0" goto init + +echo. +echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. +echo. +echo Please set the JAVA_HOME variable in your environment to match the +echo location of your Java installation. + +goto fail + +:findJavaFromJavaHome +set JAVA_HOME=%JAVA_HOME:"=% +set JAVA_EXE=%JAVA_HOME%/bin/java.exe + +if exist "%JAVA_EXE%" goto init + +echo. +echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME% +echo. +echo Please set the JAVA_HOME variable in your environment to match the +echo location of your Java installation. + +goto fail + +:init +@rem Get command-line arguments, handling Windows variants + +if not "%OS%" == "Windows_NT" goto win9xME_args + +:win9xME_args +@rem Slurp the command line arguments. +set CMD_LINE_ARGS= +set _SKIP=2 + +:win9xME_args_slurp +if "x%~1" == "x" goto execute + +set CMD_LINE_ARGS=%* + +:execute +@rem Setup the command line + +set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar + +@rem Execute Gradle +"%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %CMD_LINE_ARGS% + +:end +@rem End local scope for the variables with windows NT shell +if "%ERRORLEVEL%"=="0" goto mainEnd + +:fail +rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of +rem the _cmd.exe /c_ return code! +if not "" == "%GRADLE_EXIT_CONSOLE%" exit 1 +exit /b 1 + +:mainEnd +if "%OS%"=="Windows_NT" endlocal + +:omega diff --git a/proto/grpc_proxy/scripts/gen_protobuf.sh b/proto/grpc_proxy/scripts/gen_protobuf.sh new file mode 100644 index 000000000..d65df4efe --- /dev/null +++ b/proto/grpc_proxy/scripts/gen_protobuf.sh @@ -0,0 +1,11 @@ +#!/bin/bash +cd ../../.. +yarn gen Protobuf +if [ -d "proto/grpc_proxy/src/main/proto/sdk/4.0" ]; then + rmdir proto/grpc_proxy/src/main/proto/sdk/4.0 +fi +yarn gen GrpcProxy +if [ -d "proto/grpc_proxy/src/main/java/com/google/looker/server/sdk/4.0" ]; then + rmdir proto/grpc_proxy/src/main/java/com/google/looker/server/sdk/4.0 +fi +rm proto/grpc_proxy/src/main/java/com/google/looker/server/sdk/LookerModels.java diff --git a/proto/grpc_proxy/scripts/gen_protobuf_openapi.sh b/proto/grpc_proxy/scripts/gen_protobuf_openapi.sh new file mode 100644 index 000000000..0cf9e74c1 --- /dev/null +++ b/proto/grpc_proxy/scripts/gen_protobuf_openapi.sh @@ -0,0 +1,5 @@ +#!/bin/bash + +# Not used as not ready for prime time. Use as reference for the Looker protobuf generator. + +openapi-generator generate -g protobuf-schema -o ../src/main/proto2 -i ../../../spec/Looker.4.0.oas.json --package-name looker --additional-properties=identifierNamingConvention=snake_case diff --git a/proto/grpc_proxy/scripts/ssl_setup.sh b/proto/grpc_proxy/scripts/ssl_setup.sh new file mode 100644 index 000000000..bd1391479 --- /dev/null +++ b/proto/grpc_proxy/scripts/ssl_setup.sh @@ -0,0 +1,11 @@ +#!/bin/bash +cd .. +test ! -d ssl && mkdir ssl +cd ssl +SERVER_CN=localhost +openssl genrsa -passout pass:1111 -des3 -out ca.key 4096 +openssl req -passin pass:1111 -new -x509 -days 365 -key ca.key -out ca.crt -subj "/CN=${SERVER_CN}" +openssl genrsa -passout pass:1111 -des3 -out server.key 4096 +openssl req -passin pass:1111 -new -key server.key -out server.csr -subj "/CN=${SERVER_CN}" +openssl x509 -req -passin pass:1111 -days 365 -in server.csr -CA ca.crt -CAkey ca.key -set_serial 01 -out server.crt +openssl pkcs8 -topk8 -nocrypt -passin pass:1111 -in server.key -out server.pem diff --git a/proto/grpc_proxy/settings.gradle b/proto/grpc_proxy/settings.gradle new file mode 100644 index 000000000..096d59181 --- /dev/null +++ b/proto/grpc_proxy/settings.gradle @@ -0,0 +1,2 @@ +rootProject.name = 'grpc_proxy' + diff --git a/proto/grpc_proxy/src/main/java/com/google/looker/client/LookerGrpcClient.java b/proto/grpc_proxy/src/main/java/com/google/looker/client/LookerGrpcClient.java new file mode 100644 index 000000000..8ce929d1d --- /dev/null +++ b/proto/grpc_proxy/src/main/java/com/google/looker/client/LookerGrpcClient.java @@ -0,0 +1,150 @@ +package com.google.looker.client; + +import com.google.looker.common.BearerToken; +import com.google.looker.common.Constants; +import com.google.looker.grpc.services.AccessToken; +import com.google.looker.grpc.services.LoginReq; +import com.google.looker.grpc.services.LoginResponse; +import com.google.looker.grpc.services.LogoutReq; +import com.google.looker.grpc.services.LookerServiceGrpc; +import com.google.looker.grpc.services.LookerStreamingServiceGrpc; +import com.google.looker.grpc.services.PingServiceGrpc; +import io.github.cdimascio.dotenv.Dotenv; +import io.grpc.ManagedChannel; +import io.grpc.netty.shaded.io.grpc.netty.GrpcSslContexts; +import io.grpc.netty.shaded.io.grpc.netty.NettyChannelBuilder; +import java.io.File; +import javax.net.ssl.SSLException; +import org.apache.commons.lang3.math.NumberUtils; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +public class LookerGrpcClient { + + static { + Dotenv dotenv = Dotenv.load(); + dotenv.entries().forEach(e -> System.setProperty(e.getKey(), e.getValue())); + } + + private static final Logger LOGGER = LoggerFactory.getLogger(LookerGrpcClient.class); + + private SSLException initFailure; + private ManagedChannel channel; + private PingServiceGrpc.PingServiceBlockingStub pingBlockingStub; + private LookerServiceGrpc.LookerServiceBlockingStub lookerServiceBlockingStub; + private LookerStreamingServiceGrpc.LookerStreamingServiceStub lookerStreamingServiceStub; + private AccessToken accessTokenResult; + + public LookerGrpcClient() { + try { + channel = NettyChannelBuilder + .forAddress( + System.getProperty(Constants.GRPC_SERVER_HOST), + NumberUtils.toInt(System.getProperty(Constants.GRPC_SERVER_LISTEN_PORT)) + ) + .sslContext( + GrpcSslContexts + .forClient() + .trustManager(new File(System.getProperty(Constants.TRUST_MANAGER_FILE)) + ).build()) + .build(); + } catch (SSLException e) { + LOGGER.error("initialization failure"); + initFailure = e; + } + } + + public PingServiceGrpc.PingServiceBlockingStub getPingBlockingStub() throws SSLException { + if (initFailure != null) { + throw initFailure; + } + if (pingBlockingStub == null) { + pingBlockingStub = PingServiceGrpc + .newBlockingStub(channel); + } + return pingBlockingStub; + } + + public LookerServiceGrpc.LookerServiceBlockingStub getLookerServiceBlockingStub() throws SSLException { + if (initFailure != null) { + throw initFailure; + } + if (lookerServiceBlockingStub == null) { + if (accessTokenResult == null) { + LOGGER.debug("create blocking stub WITHOUT credentials"); + lookerServiceBlockingStub = LookerServiceGrpc + .newBlockingStub(channel); + } else { + LOGGER.debug("create blocking stub WITH credentials: " + accessTokenResult.getAccessToken()); + BearerToken token = new BearerToken(accessTokenResult.getAccessToken()); + lookerServiceBlockingStub = LookerServiceGrpc + .newBlockingStub(channel) + .withCallCredentials(token); + } + } + return lookerServiceBlockingStub; + } + + public LookerStreamingServiceGrpc.LookerStreamingServiceStub getLookerStreamingServiceStub() throws SSLException { + if (initFailure != null) { + throw initFailure; + } + if (lookerStreamingServiceStub == null) { + if (accessTokenResult == null) { + LOGGER.debug("create blocking stub WITHOUT credentials"); + lookerStreamingServiceStub = LookerStreamingServiceGrpc + .newStub(channel); + } else { + LOGGER.debug("create blocking stub WITH credentials: " + accessTokenResult.getAccessToken()); + BearerToken token = new BearerToken(accessTokenResult.getAccessToken()); + lookerStreamingServiceStub = LookerStreamingServiceGrpc + .newStub(channel) + .withCallCredentials(token); + } + } + return lookerStreamingServiceStub; + } + + public void clearAccessToken() { + accessTokenResult = null; + lookerServiceBlockingStub = null; + lookerStreamingServiceStub = null; + } + + public void login() throws SSLException { + accessTokenResult = null; + lookerServiceBlockingStub = null; + lookerStreamingServiceStub = null; + LookerServiceGrpc.LookerServiceBlockingStub stub = getLookerServiceBlockingStub(); + LoginResponse response = stub.login( + LoginReq + .newBuilder() + .setClientId(System.getProperty(Constants.LOOKER_CLIENT_ID)) + .setClientSecret(System.getProperty(Constants.LOOKER_CLIENT_SECRET)) + .build() + ); + accessTokenResult = response.getResult(); + lookerServiceBlockingStub = null; + } + + public void logout() throws SSLException { + if (accessTokenResult != null) { + LookerServiceGrpc.LookerServiceBlockingStub stub = getLookerServiceBlockingStub(); + accessTokenResult = null; + lookerServiceBlockingStub = null; + stub.logout( + LogoutReq + .newBuilder() + .build() + ); + } + } + + public String getAccessToken() { + return accessTokenResult == null ? null: accessTokenResult.getAccessToken(); + } + + public long getAccessTokenExpires() { + return accessTokenResult == null ? -1: accessTokenResult.getExpiresIn(); + } +} diff --git a/proto/grpc_proxy/src/main/java/com/google/looker/common/BearerToken.java b/proto/grpc_proxy/src/main/java/com/google/looker/common/BearerToken.java new file mode 100644 index 000000000..1478f1a59 --- /dev/null +++ b/proto/grpc_proxy/src/main/java/com/google/looker/common/BearerToken.java @@ -0,0 +1,36 @@ +package com.google.looker.common; + +import io.grpc.CallCredentials; +import io.grpc.Metadata; +import io.grpc.Status; +import java.util.concurrent.Executor; + +public class BearerToken extends CallCredentials { + + private String value; + + public BearerToken(String value) { + this.value = value; + } + + @Override + public void applyRequestMetadata(RequestInfo requestInfo, Executor executor, MetadataApplier metadataApplier) { + executor.execute(() -> { + try { + Metadata headers = new Metadata(); + headers.put( + Constants.AUTHORIZATION_METADATA_KEY, + String.format("%s %s", Constants.BEARER_TYPE, value) + ); + metadataApplier.apply(headers); + } catch (Throwable e) { + metadataApplier.fail(Status.UNAUTHENTICATED.withCause(e)); + } + }); + } + + @Override + public void thisUsesUnstableApi() { + // noop + } +} diff --git a/proto/grpc_proxy/src/main/java/com/google/looker/common/Constants.java b/proto/grpc_proxy/src/main/java/com/google/looker/common/Constants.java new file mode 100644 index 000000000..5914be10b --- /dev/null +++ b/proto/grpc_proxy/src/main/java/com/google/looker/common/Constants.java @@ -0,0 +1,21 @@ +package com.google.looker.common; + +import static io.grpc.Metadata.ASCII_STRING_MARSHALLER; + +import io.grpc.Context; +import io.grpc.Metadata; + +public class Constants { + public static final String GRPC_SERVER_HOST = "GRPC_SERVER_HOST"; + public static final String GRPC_SERVER_LISTEN_PORT = "GRPC_SERVER_LISTEN_PORT"; + public static final String LOOKER_CLIENT_ID = "LOOKER_CLIENT_ID"; + public static final String LOOKER_CLIENT_SECRET = "LOOKER_CLIENT_SECRET"; + public static final String LOOKER_BASE_URL = "LOOKER_BASE_URL"; + public static final String LOOKER_VERIFY_SSL = "LOOKER_VERIFY_SSL"; + public static final String CERT_CHAIN_FILE = "CERT_CHAIN_FILE"; + public static final String PRIVATE_KEY_FILE = "PRIVATE_KEY_FILE"; + public static final String TRUST_MANAGER_FILE = "TRUST_MANAGER_FILE"; + public static final String BEARER_TYPE = "Bearer"; + public static final Metadata.Key AUTHORIZATION_METADATA_KEY = Metadata.Key.of("Authorization", ASCII_STRING_MARSHALLER); + public static final Context.Key CLIENT_ID_CONTEXT_KEY = Context.key("clientId"); +} diff --git a/proto/grpc_proxy/src/main/java/com/google/looker/server/Server.java b/proto/grpc_proxy/src/main/java/com/google/looker/server/Server.java new file mode 100644 index 000000000..1ce4960c6 --- /dev/null +++ b/proto/grpc_proxy/src/main/java/com/google/looker/server/Server.java @@ -0,0 +1,55 @@ +package com.google.looker.server; + +import com.google.looker.common.Constants; +import com.google.looker.server.rtl.auth.AuthorizationInterceptor; +import com.google.looker.server.rtl.ping.PingServiceImpl; +import com.google.looker.server.sdk.LookerServiceImpl; +import com.google.looker.server.sdk.LookerStreamingServiceImpl; +import io.github.cdimascio.dotenv.Dotenv; +import io.grpc.ServerBuilder; +import java.io.File; +import java.io.IOException; +import org.apache.commons.lang3.math.NumberUtils; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +public class Server { + + static { + Dotenv dotenv = Dotenv.load(); + dotenv.entries().forEach(e -> System.setProperty(e.getKey(), e.getValue())); + } + + private static final Logger LOGGER = LoggerFactory.getLogger(Server.class); + + public void run() throws IOException, InterruptedException { + System.out.println("Server starting up"); + io.grpc.Server server = ServerBuilder + .forPort(NumberUtils.toInt(System.getProperty(Constants.GRPC_SERVER_LISTEN_PORT))) + .addService(new PingServiceImpl()) + .addService(new LookerServiceImpl()) + .addService(new LookerStreamingServiceImpl()) + .useTransportSecurity( + new File(System.getProperty(Constants.CERT_CHAIN_FILE)), + new File(System.getProperty(Constants.PRIVATE_KEY_FILE)) + ) + .intercept(new AuthorizationInterceptor()) + .build(); + server.start(); + LOGGER.info("Server running"); + Runtime.getRuntime().addShutdownHook(new Thread(() -> { + LOGGER.info("Server shutdown request received"); + server.shutdown(); + LOGGER.info("Server shutdown"); + })); + server.awaitTermination(); + } + + public static void main(String[] args) throws IOException, InterruptedException { + try { + new Server().run(); + } catch (Exception e) { + e.printStackTrace(); + } + } +} diff --git a/proto/grpc_proxy/src/main/java/com/google/looker/server/rtl/DefaultTransport.java b/proto/grpc_proxy/src/main/java/com/google/looker/server/rtl/DefaultTransport.java new file mode 100644 index 000000000..43b281437 --- /dev/null +++ b/proto/grpc_proxy/src/main/java/com/google/looker/server/rtl/DefaultTransport.java @@ -0,0 +1,201 @@ +package com.google.looker.server.rtl; + +import com.google.gson.Gson; +import com.google.gson.reflect.TypeToken; +import com.google.looker.common.Constants; +import io.grpc.Status; +import java.io.IOException; +import java.io.UnsupportedEncodingException; +import java.lang.reflect.Type; +import java.net.URLEncoder; +import java.security.KeyManagementException; +import java.security.NoSuchAlgorithmException; +import java.util.HashMap; +import java.util.Map; +import javax.net.ssl.SSLContext; +import javax.net.ssl.SSLSocketFactory; +import javax.net.ssl.TrustManager; +import javax.net.ssl.X509TrustManager; +import okhttp3.MediaType; +import okhttp3.OkHttpClient; +import okhttp3.Request; +import okhttp3.RequestBody; +import okhttp3.Response; +import okhttp3.ResponseBody; +import org.apache.commons.lang3.ClassUtils; +import org.apache.commons.lang3.StringUtils; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +final public class DefaultTransport implements Transport { + + final private static Logger LOGGER = LoggerFactory.getLogger(DefaultTransport.class); + + private OkHttpClient client; + + public static final MediaType JSON + = MediaType.parse("application/json; charset=utf-8"); + + public LookerClientResponse request( + String apiVersion, + HttpMethod method, + String path, + String inputJson) { + String accessToken = Constants.CLIENT_ID_CONTEXT_KEY.get(); + if (accessToken == null || StringUtils.isBlank(accessToken)) { + LOGGER.debug("request ignored because no access token"); + return new LookerClientResponse(Status.NOT_FOUND); + } else { + Gson gson = new Gson(); + Type inputDataMapType = new TypeToken>() { + }.getType(); + Map inputData = gson.fromJson(inputJson, inputDataMapType); + String fullPath = makePath(apiVersion, path, inputData); + Request.Builder requestBuilder = new Request.Builder() + .url(fullPath) + .addHeader("Content-Type", "application/json") + .addHeader("Authorization", "Bearer " + accessToken) + .addHeader("x-looker-appid", "Looker GRPC Proxy Server"); + addMethod(requestBuilder, method, inputData); + Request request = requestBuilder.build(); + try { + Response response = getHttpClient().newCall(request).execute(); + int statusCode = response.code(); + if (statusCode > 199 && statusCode < 300) { + // TODO do not assume json + ResponseBody responseBody = response.body(); + if (responseBody == null) { + LOGGER.error("response has no body"); + return new LookerClientResponse(Status.NOT_FOUND); + } else { + String lookerResponse = responseBody.string(); + if (!(lookerResponse.startsWith("{") || lookerResponse.startsWith("["))) { + // TODO handle number or boolean instead of assuming string + lookerResponse = "\"" + lookerResponse + "\""; + } + String defaultResponse = String.format("{\"result\":%s}", lookerResponse); + LOGGER.debug("request succeeded " + defaultResponse); + return new LookerClientResponse(statusCode, defaultResponse); + } + } else { + LOGGER.debug("request failed: " + statusCode); + return new LookerClientResponse(statusCode); + } + } catch (IOException | KeyManagementException | NoSuchAlgorithmException e) { + LOGGER.error("login request failed", e); + return new LookerClientResponse(Status.INTERNAL); + } + } + } + + final public OkHttpClient getHttpClient() + throws NoSuchAlgorithmException, KeyManagementException { + if (this.client == null) { + OkHttpClient.Builder builder = new OkHttpClient.Builder(); + if (System.getProperty(Constants.LOOKER_VERIFY_SSL).equals("false")) { + final TrustManager[] trustAllCerts = new TrustManager[]{ + new X509TrustManager() { + + @Override + public void checkClientTrusted(java.security.cert.X509Certificate[] chain, + String authType) { + } + + @Override + public void checkServerTrusted(java.security.cert.X509Certificate[] chain, + String authType) { + } + + @Override + public java.security.cert.X509Certificate[] getAcceptedIssuers() { + return new java.security.cert.X509Certificate[]{}; + } + } + }; + final SSLContext sslContext = SSLContext.getInstance("SSL"); + sslContext.init(null, trustAllCerts, new java.security.SecureRandom()); + final SSLSocketFactory sslSocketFactory = sslContext.getSocketFactory(); + builder.sslSocketFactory(sslSocketFactory, (X509TrustManager) trustAllCerts[0]) + .hostnameVerifier((hostname, session) -> true) + ; + } + this.client = builder.build(); + } + return this.client; + } + + public String makePath(String apiVersion, String path, Map inputData) { + String fullPath = String.format("%s/api/%s%s", + System.getProperty(Constants.LOOKER_BASE_URL), + apiVersion, + updatePath(path, inputData) + ); + LOGGER.debug("fullpath=" + fullPath); + return fullPath; + } + + private String updatePath(String path, Map inputData) { + Map qsMap = new HashMap<>(); + StringBuilder updatedPath = new StringBuilder(path); + for (Map.Entry entry : inputData.entrySet()) { + String key = entry.getKey(); + Object value = entry.getValue(); + // TODO the handling of the body is potentially brittle. Modify generator to make it not so. + if (!key.equals("body") && + value != null && + (value instanceof String || ClassUtils.isPrimitiveOrWrapper(value.getClass()))) { + String searchValue = "{" + key + "}"; + if (StringUtils.contains(updatedPath.toString(), searchValue)) { + updatedPath = new StringBuilder(StringUtils + .replace(updatedPath.toString(), searchValue, value.toString())); + } else { + qsMap.put(entry.getKey(), entry.getValue().toString()); + } + } + } + if (qsMap.size() > 0) { + String sep = "?"; + for (Map.Entry entry : qsMap.entrySet()) { + updatedPath.append(sep); + updatedPath.append(entry.getKey()).append("=").append(encodeValue(entry.getValue())); + sep = "&"; + } + } + return updatedPath.toString(); + } + + private String encodeValue(String value) { + try { + return URLEncoder.encode(value, "UTF8"); + } catch (UnsupportedEncodingException e) { + return value; + } + } + + private void addMethod(Request.Builder requestBuilder, HttpMethod method, Map inputData) { + switch (method) { + case GET: + requestBuilder.get(); + break; + case POST: + requestBuilder.post(createRequestBody(inputData.get("body"))); + break; + case PUT: + requestBuilder.put(createRequestBody(inputData.get("body"))); + break; + case PATCH: + requestBuilder.patch(createRequestBody(inputData.get("body"))); + break; + case DELETE: + requestBuilder.delete(createRequestBody(inputData.get("body"))); + break; + } + } + + private RequestBody createRequestBody(Object body) { + Gson gson = new Gson(); + String jsonBody = body == null ? "{}" : gson.toJson(body); + return RequestBody.create(jsonBody, MediaType.parse("application/json")); + } + +} diff --git a/proto/grpc_proxy/src/main/java/com/google/looker/server/rtl/LoginTransport.java b/proto/grpc_proxy/src/main/java/com/google/looker/server/rtl/LoginTransport.java new file mode 100644 index 000000000..48e779228 --- /dev/null +++ b/proto/grpc_proxy/src/main/java/com/google/looker/server/rtl/LoginTransport.java @@ -0,0 +1,71 @@ +package com.google.looker.server.rtl; + +import com.google.gson.Gson; +import com.google.gson.reflect.TypeToken; +import com.google.looker.common.Constants; +import io.grpc.Status; +import java.io.IOException; +import java.lang.reflect.Type; +import java.security.KeyManagementException; +import java.security.NoSuchAlgorithmException; +import java.util.Map; +import okhttp3.FormBody; +import okhttp3.OkHttpClient; +import okhttp3.Request; +import okhttp3.Response; +import okhttp3.ResponseBody; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +public class LoginTransport implements Transport { + + final private static Logger LOGGER = LoggerFactory.getLogger(LoginTransport.class); + + public LookerClientResponse request( + String apiVersion, + HttpMethod method, + String path, + String inputJson) { + LOGGER.debug(inputJson); + Gson gson = new Gson(); + Type inputDataMapType = new TypeToken>() {}.getType(); + Map inputData = gson.fromJson(inputJson, inputDataMapType); + FormBody.Builder builder = new FormBody.Builder(); + inputData.forEach((k, v) -> builder.add(k, (String) v)); + String fullPath = String.format("%s/api/%s%s", System.getProperty(Constants.LOOKER_BASE_URL), apiVersion, path); + LOGGER.debug("fullpath=" + fullPath); + Request request = new Request.Builder() + .url(fullPath) + .addHeader("Content-Type", "application/x-www-form-urlencoded; charset=UTF-8") + .addHeader("x-looker-appid", "Looker GRPC Proxy Server") + .post(builder.build()) + .build(); + try { + Response response = getHttpClient().newCall(request).execute(); + int statusCode = response.code(); + if (statusCode > 199 && statusCode < 300) { + ResponseBody responseBody = response.body(); + if (responseBody == null) { + LOGGER.error("login response has no body"); + return new LookerClientResponse(Status.NOT_FOUND); + } else { + String loginResponse = String.format("{\"result\":%s}", responseBody.string()); + LOGGER.debug("login request succeeded " + loginResponse); + return new LookerClientResponse(statusCode, loginResponse); + } + } else { + LOGGER.debug("login request failed: " + statusCode); + return new LookerClientResponse(statusCode); + } + } catch (IOException | KeyManagementException | NoSuchAlgorithmException e) { + LOGGER.error("login request failed", e); + return new LookerClientResponse(Status.INTERNAL); + } + } + + @Override + public OkHttpClient getHttpClient() throws KeyManagementException, NoSuchAlgorithmException { + return TransportFactory.instance().getDefaultTransport().getHttpClient(); + } + +} diff --git a/proto/grpc_proxy/src/main/java/com/google/looker/server/rtl/LogoutTransport.java b/proto/grpc_proxy/src/main/java/com/google/looker/server/rtl/LogoutTransport.java new file mode 100644 index 000000000..20146c81b --- /dev/null +++ b/proto/grpc_proxy/src/main/java/com/google/looker/server/rtl/LogoutTransport.java @@ -0,0 +1,71 @@ +package com.google.looker.server.rtl; + +import com.google.looker.common.Constants; +import io.grpc.Status; +import java.io.IOException; +import java.security.KeyManagementException; +import java.security.NoSuchAlgorithmException; +import okhttp3.OkHttpClient; +import okhttp3.Request; +import okhttp3.Response; +import okhttp3.ResponseBody; +import org.apache.commons.lang3.StringUtils; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +public class LogoutTransport implements Transport { + + final private static Logger LOGGER = LoggerFactory.getLogger(LogoutTransport.class); + + public LookerClientResponse request( + String apiVersion, + HttpMethod method, + String path, + String inputJson) { + String accessToken = Constants.CLIENT_ID_CONTEXT_KEY.get(); + if (accessToken == null || StringUtils.isBlank(accessToken)) { + LOGGER.debug("logout request ignored because no access token"); + // But we dont care + return new LookerClientResponse(200); + } else { + LOGGER.debug(inputJson); + String fullPath = String.format("%s/api/%s%s", System.getProperty(Constants.LOOKER_BASE_URL), apiVersion, path); + LOGGER.debug("fullpath=" + fullPath); + Request request = new Request.Builder() + .addHeader("Content-Type", "application/json") + .addHeader("Authorization", "Bearer " + accessToken) + .addHeader("x-looker-appid", "Looker GRPC Proxy Server") + .url(fullPath) + .delete() + .build(); + try { + Response response = getHttpClient().newCall(request).execute(); + int statusCode = response.code(); + if (statusCode > 199 && statusCode < 300) { + ResponseBody responseBody = response.body(); + LOGGER.debug("logout request succeeded"); + if (responseBody == null) { + return new LookerClientResponse(statusCode); + } else { + String logoutResponse = String.format("{\"result\":\"%s\"}", responseBody.string()); + return new LookerClientResponse(statusCode, logoutResponse); + } + } else { + LOGGER.debug("logout request failed: " + statusCode); + // But we dont care + return new LookerClientResponse(200); + } + } catch (IOException | KeyManagementException | NoSuchAlgorithmException e) { + LOGGER.error("logout request failed", e); + return new LookerClientResponse(Status.INTERNAL); + } + + } + } + + @Override + public OkHttpClient getHttpClient() throws KeyManagementException, NoSuchAlgorithmException { + return TransportFactory.instance().getDefaultTransport().getHttpClient(); + } + +} diff --git a/proto/grpc_proxy/src/main/java/com/google/looker/server/rtl/LookerClient.java b/proto/grpc_proxy/src/main/java/com/google/looker/server/rtl/LookerClient.java new file mode 100644 index 000000000..19d19a567 --- /dev/null +++ b/proto/grpc_proxy/src/main/java/com/google/looker/server/rtl/LookerClient.java @@ -0,0 +1,30 @@ +package com.google.looker.server.rtl; + +public class LookerClient { + final private String apiVersion; + + public LookerClient(String apiVersion) { + this.apiVersion = apiVersion; + } + + public LookerClientResponse get(String path, String inputJson) { + return TransportFactory.instance().getTransport(path).request(apiVersion, Transport.HttpMethod.GET, path, inputJson); + } + + public LookerClientResponse post(String path, String inputJson) { + return TransportFactory.instance().getTransport(path).request(apiVersion, Transport.HttpMethod.POST, path, inputJson); + } + + public LookerClientResponse put(String path, String inputJson) { + return TransportFactory.instance().getTransport(path).request(apiVersion, Transport.HttpMethod.PUT, path, inputJson); + } + + public LookerClientResponse patch(String path, String inputJson) { + return TransportFactory.instance().getTransport(path).request(apiVersion, Transport.HttpMethod.PATCH, path, inputJson); + } + + public LookerClientResponse delete(String path, String inputJson) { + return TransportFactory.instance().getTransport(path).request(apiVersion, Transport.HttpMethod.DELETE, path, inputJson); + } + +} diff --git a/proto/grpc_proxy/src/main/java/com/google/looker/server/rtl/LookerClientResponse.java b/proto/grpc_proxy/src/main/java/com/google/looker/server/rtl/LookerClientResponse.java new file mode 100644 index 000000000..237e4ef42 --- /dev/null +++ b/proto/grpc_proxy/src/main/java/com/google/looker/server/rtl/LookerClientResponse.java @@ -0,0 +1,43 @@ +package com.google.looker.server.rtl; + +import io.grpc.Status; + +public class LookerClientResponse { + + private Status status; + private boolean success; + private String jsonResponse; + + public LookerClientResponse(Status status) { + this.status = status; + } + + public LookerClientResponse(int statusCode,String jsonResponse) { + generateStatus(statusCode); + this.jsonResponse = jsonResponse; + } + + public LookerClientResponse(int statusCode) { + generateStatus(statusCode); + } + + public Status getStatus() { + return status; + } + + public String getJsonResponse() { + return jsonResponse; + } + + private void generateStatus(int statusCode) { + if (statusCode < 200) { + status = Status.INTERNAL; + } else if (statusCode > 299) { + if (statusCode == 404 || statusCode == 401 || statusCode == 403) { + status = Status.NOT_FOUND; + } else { + status = Status.INTERNAL; + } + } + } +} diff --git a/proto/grpc_proxy/src/main/java/com/google/looker/server/rtl/Transport.java b/proto/grpc_proxy/src/main/java/com/google/looker/server/rtl/Transport.java new file mode 100644 index 000000000..f24a5ed00 --- /dev/null +++ b/proto/grpc_proxy/src/main/java/com/google/looker/server/rtl/Transport.java @@ -0,0 +1,22 @@ +package com.google.looker.server.rtl; + +import java.security.KeyManagementException; +import java.security.NoSuchAlgorithmException; +import okhttp3.OkHttpClient; + +public interface Transport { + enum HttpMethod { + GET, + POST, + PUT, + PATCH, + DELETE + } + LookerClientResponse request( + String apiVersion, + HttpMethod method, + String path, + String inputJson); + + OkHttpClient getHttpClient() throws NoSuchAlgorithmException, KeyManagementException; +} diff --git a/proto/grpc_proxy/src/main/java/com/google/looker/server/rtl/TransportFactory.java b/proto/grpc_proxy/src/main/java/com/google/looker/server/rtl/TransportFactory.java new file mode 100644 index 000000000..eefd75aed --- /dev/null +++ b/proto/grpc_proxy/src/main/java/com/google/looker/server/rtl/TransportFactory.java @@ -0,0 +1,35 @@ +package com.google.looker.server.rtl; + +public class TransportFactory { + + private static TransportFactory instance = new TransportFactory(); + + public static TransportFactory instance() { + return instance; + } + + final private Transport defaultTransport; + final private Transport loginTransport; + final private Transport logoutTransport; + + private TransportFactory() { + defaultTransport = new DefaultTransport(); + loginTransport = new LoginTransport(); + logoutTransport = new LogoutTransport(); + } + + public Transport getDefaultTransport() { + return defaultTransport; + } + + public Transport getTransport(String path) { + if (path.startsWith("/login")) { + return loginTransport; + } else if (path.startsWith("/logout")) { + return logoutTransport; + } else { + return defaultTransport; + } + } + +} diff --git a/proto/grpc_proxy/src/main/java/com/google/looker/server/rtl/auth/AuthorizationInterceptor.java b/proto/grpc_proxy/src/main/java/com/google/looker/server/rtl/auth/AuthorizationInterceptor.java new file mode 100644 index 000000000..fcd975a71 --- /dev/null +++ b/proto/grpc_proxy/src/main/java/com/google/looker/server/rtl/auth/AuthorizationInterceptor.java @@ -0,0 +1,58 @@ +package com.google.looker.server.rtl.auth; + +import com.google.looker.common.Constants; +import io.grpc.Context; +import io.grpc.Contexts; +import io.grpc.Metadata; +import io.grpc.ServerCall; +import io.grpc.ServerCallHandler; +import io.grpc.ServerInterceptor; +import io.grpc.Status; +import java.util.Arrays; +import java.util.HashSet; +import java.util.Set; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +public class AuthorizationInterceptor implements ServerInterceptor { + + final private static Logger LOGGER = LoggerFactory.getLogger(AuthorizationInterceptor.class); + + private final Set unsecuredMethods + = new HashSet<>(Arrays.asList("Login", "LoginUser", "Ping")); + + public AuthorizationInterceptor() { + } + + @Override + public ServerCall.Listener interceptCall(ServerCall serverCall, Metadata metadata, ServerCallHandler serverCallHandler) { + String value = metadata.get(Constants.AUTHORIZATION_METADATA_KEY); + LOGGER.debug("AUTHORIZATION_METADATA_KEY=" + value); + LOGGER.info(String.format("Method called is %s", serverCall.getMethodDescriptor().getBareMethodName())); + Status status; + if (value == null) { + if (unsecuredMethods.contains(serverCall.getMethodDescriptor().getBareMethodName())) { + Context ctx = Context.current(); + return Contexts.interceptCall(ctx, serverCall, metadata, serverCallHandler); + } else { + status = Status.UNAUTHENTICATED.withDescription("Authorization token is missing"); + } + } else if (!value.startsWith(Constants.BEARER_TYPE)) { + status = Status.UNAUTHENTICATED.withDescription("Unknown authorization type"); + } else { + try { + String token = value.substring(Constants.BEARER_TYPE.length()).trim(); + LOGGER.debug("bearer token=" + token); + Context ctx = Context.current().withValue(Constants.CLIENT_ID_CONTEXT_KEY, token); + return Contexts.interceptCall(ctx, serverCall, metadata, serverCallHandler); + } catch (Exception e) { + status = Status.UNAUTHENTICATED.withDescription(e.getMessage()).withCause(e); + } + } + + serverCall.close(status, metadata); + return new ServerCall.Listener() { + // noop + }; + } +} diff --git a/proto/grpc_proxy/src/main/java/com/google/looker/server/rtl/ping/PingServiceImpl.java b/proto/grpc_proxy/src/main/java/com/google/looker/server/rtl/ping/PingServiceImpl.java new file mode 100644 index 000000000..9257da0b5 --- /dev/null +++ b/proto/grpc_proxy/src/main/java/com/google/looker/server/rtl/ping/PingServiceImpl.java @@ -0,0 +1,15 @@ +package com.google.looker.server.rtl.ping; + +import com.google.looker.grpc.services.PingServiceGrpc.PingServiceImplBase; +import com.google.looker.server.rtl.PingRequest; +import com.google.looker.server.rtl.PingResponse; +import io.grpc.stub.StreamObserver; + +public class PingServiceImpl extends PingServiceImplBase { + + @Override + public void ping(PingRequest request, StreamObserver responseObserver) { + responseObserver.onNext(PingResponse.newBuilder().setActive(true).build()); + responseObserver.onCompleted(); + } +} diff --git a/proto/grpc_proxy/src/main/java/com/google/looker/server/sdk/LookerServiceImpl.java b/proto/grpc_proxy/src/main/java/com/google/looker/server/sdk/LookerServiceImpl.java new file mode 100644 index 000000000..b8966e78c --- /dev/null +++ b/proto/grpc_proxy/src/main/java/com/google/looker/server/sdk/LookerServiceImpl.java @@ -0,0 +1,15967 @@ +// MIT License +// +// Copyright (c) 2021 Looker Data Sciences, Inc. +// +// Permission is hereby granted, free of charge, to any person obtaining a copy +// of this software and associated documentation files (the "Software"), to deal +// in the Software without restriction, including without limitation the rights +// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +// copies of the Software, and to permit persons to whom the Software is +// furnished to do so, subject to the following conditions: +// +// The above copyright notice and this permission notice shall be included in all +// copies or substantial portions of the Software. +// +// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +// SOFTWARE. +// + +// 438 API methods + + +package com.google.looker.server.sdk; + +import com.google.looker.grpc.services.*; +import com.google.looker.grpc.services.LookerServiceGrpc.LookerServiceImplBase; +import com.google.looker.server.rtl.LookerClient; +import com.google.looker.server.rtl.LookerClientResponse; +import com.google.protobuf.InvalidProtocolBufferException; +import com.google.protobuf.util.JsonFormat; +import io.grpc.Status; +import io.grpc.stub.StreamObserver; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +public class LookerServiceImpl extends LookerServiceImplBase { + + final private static Logger LOGGER = LoggerFactory.getLogger(LookerServiceImpl.class); + + final private LookerClient lookerClient; + + public LookerServiceImpl() { + lookerClient = new LookerClient("4.0"); + } + + + //#region Alert: Alert + + /** + * ### Search Alerts + * + */ + @Override + public void searchAlerts(SearchAlertsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/alerts/search", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SearchAlertsResponse.Builder responseBuilder = SearchAlertsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get an alert by a given alert ID + * + */ + @Override + public void getAlert(GetAlertReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/alerts/{alert_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + GetAlertResponse.Builder responseBuilder = GetAlertResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update an alert + * # Required fields: `owner_id`, `field`, `destinations`, `comparison_type`, `threshold`, `cron` + * # + * + */ + @Override + public void updateAlert(UpdateAlertReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.put("/alerts/{alert_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateAlertResponse.Builder responseBuilder = UpdateAlertResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update select alert fields + * # Available fields: `owner_id`, `is_disabled`, `disabled_reason`, `is_public`, `threshold` + * # + * + */ + @Override + public void updateAlertField(UpdateAlertFieldReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/alerts/{alert_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateAlertFieldResponse.Builder responseBuilder = UpdateAlertFieldResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Delete an alert by a given alert ID + * + */ + @Override + public void deleteAlert(DeleteAlertReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/alerts/{alert_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteAlertResponse.Builder responseBuilder = DeleteAlertResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create a new alert and return details of the newly created object + * + * Required fields: `field`, `destinations`, `comparison_type`, `threshold`, `cron` + * + * Example Request: + * Run alert on dashboard element '103' at 5am every day. Send an email to 'test@test.com' if inventory for Los Angeles (using dashboard filter `Warehouse Name`) is lower than 1,000 + * ``` + * { + * "cron": "0 5 * * *", + * "custom_title": "Alert when LA inventory is low", + * "dashboard_element_id": 103, + * "applied_dashboard_filters": [ + * { + * "filter_title": "Warehouse Name", + * "field_name": "distribution_centers.name", + * "filter_value": "Los Angeles CA", + * "filter_description": "is Los Angeles CA" + * } + * ], + * "comparison_type": "LESS_THAN", + * "destinations": [ + * { + * "destination_type": "EMAIL", + * "email_address": "test@test.com" + * } + * ], + * "field": { + * "title": "Number on Hand", + * "name": "inventory_items.number_on_hand" + * }, + * "is_disabled": false, + * "is_public": true, + * "threshold": 1000 + * } + * ``` + * + */ + @Override + public void createAlert(CreateAlertReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/alerts", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateAlertResponse.Builder responseBuilder = CreateAlertResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Enqueue an Alert by ID + * + */ + @Override + public void enqueueAlert(EnqueueAlertReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/alerts/{alert_id}/enqueue", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + EnqueueAlertResponse.Builder responseBuilder = EnqueueAlertResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + //#endregion Alert: Alert + + //#region ApiAuth: API Authentication + + /** + * ### Present client credentials to obtain an authorization token + * + * Looker API implements the OAuth2 [Resource Owner Password Credentials Grant](https://looker.com/docs/r/api/outh2_resource_owner_pc) pattern. + * The client credentials required for this login must be obtained by creating an API3 key on a user account + * in the Looker Admin console. The API3 key consists of a public `client_id` and a private `client_secret`. + * + * The access token returned by `login` must be used in the HTTP Authorization header of subsequent + * API requests, like this: + * ``` + * Authorization: token 4QDkCyCtZzYgj4C2p2cj3csJH7zqS5RzKs2kTnG4 + * ``` + * Replace "4QDkCy..." with the `access_token` value returned by `login`. + * The word `token` is a string literal and must be included exactly as shown. + * + * This function can accept `client_id` and `client_secret` parameters as URL query params or as www-form-urlencoded params in the body of the HTTP request. Since there is a small risk that URL parameters may be visible to intermediate nodes on the network route (proxies, routers, etc), passing credentials in the body of the request is considered more secure than URL params. + * + * Example of passing credentials in the HTTP request body: + * ```` + * POST HTTP /login + * Content-Type: application/x-www-form-urlencoded + * + * client_id=CGc9B7v7J48dQSJvxxx&client_secret=nNVS9cSS3xNpSC9JdsBvvvvv + * ```` + * + * ### Best Practice: + * Always pass credentials in body params. Pass credentials in URL query params **only** when you cannot pass body params due to application, tool, or other limitations. + * + * For more information and detailed examples of Looker API authorization, see [How to Authenticate to Looker API3](https://github.com/looker/looker-sdk-ruby/blob/master/authentication.md). + * + */ + @Override + public void login(LoginReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/login", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + LoginResponse.Builder responseBuilder = LoginResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create an access token that runs as a given user. + * + * This can only be called by an authenticated admin user. It allows that admin to generate a new + * authentication token for the user with the given user id. That token can then be used for subsequent + * API calls - which are then performed *as* that target user. + * + * The target user does *not* need to have a pre-existing API client_id/client_secret pair. And, no such + * credentials are created by this call. + * + * This allows for building systems where api user authentication for an arbitrary number of users is done + * outside of Looker and funneled through a single 'service account' with admin permissions. Note that a + * new access token is generated on each call. If target users are going to be making numerous API + * calls in a short period then it is wise to cache this authentication token rather than call this before + * each of those API calls. + * + * See 'login' for more detail on the access token and how to use it. + * + */ + @Override + public void loginUser(LoginUserReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/login/{user_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + LoginUserResponse.Builder responseBuilder = LoginUserResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Logout of the API and invalidate the current access token. + * + */ + @Override + public void logout(LogoutReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/logout", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + LogoutResponse.Builder responseBuilder = LogoutResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + //#endregion ApiAuth: API Authentication + + //#region Auth: Manage User Authentication Configuration + + /** + * ### Create an embed secret using the specified information. + * + * The value of the `secret` field will be set by Looker and returned. + * + */ + @Override + public void createEmbedSecret(CreateEmbedSecretReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/embed_config/secrets", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateEmbedSecretResponse.Builder responseBuilder = CreateEmbedSecretResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Delete an embed secret. + * + */ + @Override + public void deleteEmbedSecret(DeleteEmbedSecretReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/embed_config/secrets/{embed_secret_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteEmbedSecretResponse.Builder responseBuilder = DeleteEmbedSecretResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create SSO Embed URL + * + * Creates an SSO embed URL and cryptographically signs it with an embed secret. + * This signed URL can then be used to instantiate a Looker embed session in a PBL web application. + * Do not make any modifications to this URL - any change may invalidate the signature and + * cause the URL to fail to load a Looker embed session. + * + * A signed SSO embed URL can only be used once. After it has been used to request a page from the + * Looker server, the URL is invalid. Future requests using the same URL will fail. This is to prevent + * 'replay attacks'. + * + * The `target_url` property must be a complete URL of a Looker UI page - scheme, hostname, path and query params. + * To load a dashboard with id 56 and with a filter of `Date=1 years`, the looker URL would look like `https:/myname.looker.com/dashboards/56?Date=1%20years`. + * The best way to obtain this target_url is to navigate to the desired Looker page in your web browser, + * copy the URL shown in the browser address bar and paste it into the `target_url` property as a quoted string value in this API request. + * + * Permissions for the embed user are defined by the groups in which the embed user is a member (group_ids property) + * and the lists of models and permissions assigned to the embed user. + * At a minimum, you must provide values for either the group_ids property, or both the models and permissions properties. + * These properties are additive; an embed user can be a member of certain groups AND be granted access to models and permissions. + * + * The embed user's access is the union of permissions granted by the group_ids, models, and permissions properties. + * + * This function does not strictly require all group_ids, user attribute names, or model names to exist at the moment the + * SSO embed url is created. Unknown group_id, user attribute names or model names will be passed through to the output URL. + * To diagnose potential problems with an SSO embed URL, you can copy the signed URL into the Embed URI Validator text box in `/admin/embed`. + * + * The `secret_id` parameter is optional. If specified, its value must be the id of an active secret defined in the Looker instance. + * if not specified, the URL will be signed using the newest active secret defined in the Looker instance. + * + * #### Security Note + * Protect this signed URL as you would an access token or password credentials - do not write + * it to disk, do not pass it to a third party, and only pass it through a secure HTTPS + * encrypted transport. + * + */ + @Override + public void createSsoEmbedUrl(CreateSsoEmbedUrlReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/embed/sso_url", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateSsoEmbedUrlResponse.Builder responseBuilder = CreateSsoEmbedUrlResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create an Embed URL + * + * Creates an embed URL that runs as the Looker user making this API call. ("Embed as me") + * This embed URL can then be used to instantiate a Looker embed session in a + * "Powered by Looker" (PBL) web application. + * + * This is similar to Private Embedding (https://docs.looker.com/r/admin/embed/private-embed). Instead of + * of logging into the Web UI to authenticate, the user has already authenticated against the API to be able to + * make this call. However, unlike Private Embed where the user has access to any other part of the Looker UI, + * the embed web session created by requesting the EmbedUrlResponse.url in a browser only has access to + * content visible under the `/embed` context. + * + * An embed URL can only be used once, and must be used within 5 minutes of being created. After it + * has been used to request a page from the Looker server, the URL is invalid. Future requests using + * the same URL will fail. This is to prevent 'replay attacks'. + * + * The `target_url` property must be a complete URL of a Looker Embedded UI page - scheme, hostname, path starting with "/embed" and query params. + * To load a dashboard with id 56 and with a filter of `Date=1 years`, the looker Embed URL would look like `https://myname.looker.com/embed/dashboards/56?Date=1%20years`. + * The best way to obtain this target_url is to navigate to the desired Looker page in your web browser, + * copy the URL shown in the browser address bar, insert "/embed" after the host/port, and paste it into the `target_url` property as a quoted string value in this API request. + * + * #### Security Note + * Protect this embed URL as you would an access token or password credentials - do not write + * it to disk, do not pass it to a third party, and only pass it through a secure HTTPS + * encrypted transport. + * + */ + @Override + public void createEmbedUrlAsMe(CreateEmbedUrlAsMeReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/embed/token_url/me", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateEmbedUrlAsMeResponse.Builder responseBuilder = CreateEmbedUrlAsMeResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get the LDAP configuration. + * + * Looker can be optionally configured to authenticate users against an Active Directory or other LDAP directory server. + * LDAP setup requires coordination with an administrator of that directory server. + * + * Only Looker administrators can read and update the LDAP configuration. + * + * Configuring LDAP impacts authentication for all users. This configuration should be done carefully. + * + * Looker maintains a single LDAP configuration. It can be read and updated. Updates only succeed if the new state will be valid (in the sense that all required fields are populated); it is up to you to ensure that the configuration is appropriate and correct). + * + * LDAP is enabled or disabled for Looker using the **enabled** field. + * + * Looker will never return an **auth_password** field. That value can be set, but never retrieved. + * + * See the [Looker LDAP docs](https://www.looker.com/docs/r/api/ldap_setup) for additional information. + * + */ + @Override + public void ldapConfig(LdapConfigReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/ldap_config", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + LdapConfigResponse.Builder responseBuilder = LdapConfigResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update the LDAP configuration. + * + * Configuring LDAP impacts authentication for all users. This configuration should be done carefully. + * + * Only Looker administrators can read and update the LDAP configuration. + * + * LDAP is enabled or disabled for Looker using the **enabled** field. + * + * It is **highly** recommended that any LDAP setting changes be tested using the APIs below before being set globally. + * + * See the [Looker LDAP docs](https://www.looker.com/docs/r/api/ldap_setup) for additional information. + * + */ + @Override + public void updateLdapConfig(UpdateLdapConfigReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/ldap_config", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateLdapConfigResponse.Builder responseBuilder = UpdateLdapConfigResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Test the connection settings for an LDAP configuration. + * + * This tests that the connection is possible given a connection_host and connection_port. + * + * **connection_host** and **connection_port** are required. **connection_tls** is optional. + * + * Example: + * ```json + * { + * "connection_host": "ldap.example.com", + * "connection_port": "636", + * "connection_tls": true + * } + * ``` + * + * No authentication to the LDAP server is attempted. + * + * The active LDAP settings are not modified. + * + */ + @Override + public void testLdapConfigConnection(TestLdapConfigConnectionReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.put("/ldap_config/test_connection", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + TestLdapConfigConnectionResponse.Builder responseBuilder = TestLdapConfigConnectionResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Test the connection authentication settings for an LDAP configuration. + * + * This tests that the connection is possible and that a 'server' account to be used by Looker can authenticate to the LDAP server given connection and authentication information. + * + * **connection_host**, **connection_port**, and **auth_username**, are required. **connection_tls** and **auth_password** are optional. + * + * Example: + * ```json + * { + * "connection_host": "ldap.example.com", + * "connection_port": "636", + * "connection_tls": true, + * "auth_username": "cn=looker,dc=example,dc=com", + * "auth_password": "secret" + * } + * ``` + * + * Looker will never return an **auth_password**. If this request omits the **auth_password** field, then the **auth_password** value from the active config (if present) will be used for the test. + * + * The active LDAP settings are not modified. + * + * + */ + @Override + public void testLdapConfigAuth(TestLdapConfigAuthReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.put("/ldap_config/test_auth", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + TestLdapConfigAuthResponse.Builder responseBuilder = TestLdapConfigAuthResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Test the user authentication settings for an LDAP configuration without authenticating the user. + * + * This test will let you easily test the mapping for user properties and roles for any user without needing to authenticate as that user. + * + * This test accepts a full LDAP configuration along with a username and attempts to find the full info for the user from the LDAP server without actually authenticating the user. So, user password is not required.The configuration is validated before attempting to contact the server. + * + * **test_ldap_user** is required. + * + * The active LDAP settings are not modified. + * + * + */ + @Override + public void testLdapConfigUserInfo(TestLdapConfigUserInfoReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.put("/ldap_config/test_user_info", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + TestLdapConfigUserInfoResponse.Builder responseBuilder = TestLdapConfigUserInfoResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Test the user authentication settings for an LDAP configuration. + * + * This test accepts a full LDAP configuration along with a username/password pair and attempts to authenticate the user with the LDAP server. The configuration is validated before attempting the authentication. + * + * Looker will never return an **auth_password**. If this request omits the **auth_password** field, then the **auth_password** value from the active config (if present) will be used for the test. + * + * **test_ldap_user** and **test_ldap_password** are required. + * + * The active LDAP settings are not modified. + * + * + */ + @Override + public void testLdapConfigUserAuth(TestLdapConfigUserAuthReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.put("/ldap_config/test_user_auth", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + TestLdapConfigUserAuthResponse.Builder responseBuilder = TestLdapConfigUserAuthResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### List All OAuth Client Apps + * + * Lists all applications registered to use OAuth2 login with this Looker instance, including + * enabled and disabled apps. + * + * Results are filtered to include only the apps that the caller (current user) + * has permission to see. + * + */ + @Override + public void allOauthClientApps(AllOauthClientAppsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/oauth_client_apps", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllOauthClientAppsResponse.Builder responseBuilder = AllOauthClientAppsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get Oauth Client App + * + * Returns the registered app client with matching client_guid. + * + */ + @Override + public void oauthClientApp(OauthClientAppReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/oauth_client_apps/{client_guid}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + OauthClientAppResponse.Builder responseBuilder = OauthClientAppResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Register an OAuth2 Client App + * + * Registers details identifying an external web app or native app as an OAuth2 login client of the Looker instance. + * The app registration must provide a unique client_guid and redirect_uri that the app will present + * in OAuth login requests. If the client_guid and redirect_uri parameters in the login request do not match + * the app details registered with the Looker instance, the request is assumed to be a forgery and is rejected. + * + */ + @Override + public void registerOauthClientApp(RegisterOauthClientAppReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/oauth_client_apps/{client_guid}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + RegisterOauthClientAppResponse.Builder responseBuilder = RegisterOauthClientAppResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update OAuth2 Client App Details + * + * Modifies the details a previously registered OAuth2 login client app. + * + */ + @Override + public void updateOauthClientApp(UpdateOauthClientAppReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/oauth_client_apps/{client_guid}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateOauthClientAppResponse.Builder responseBuilder = UpdateOauthClientAppResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Delete OAuth Client App + * + * Deletes the registration info of the app with the matching client_guid. + * All active sessions and tokens issued for this app will immediately become invalid. + * + * ### Note: this deletion cannot be undone. + * + */ + @Override + public void deleteOauthClientApp(DeleteOauthClientAppReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/oauth_client_apps/{client_guid}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteOauthClientAppResponse.Builder responseBuilder = DeleteOauthClientAppResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Invalidate All Issued Tokens + * + * Immediately invalidates all auth codes, sessions, access tokens and refresh tokens issued for + * this app for ALL USERS of this app. + * + */ + @Override + public void invalidateTokens(InvalidateTokensReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/oauth_client_apps/{client_guid}/tokens", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + InvalidateTokensResponse.Builder responseBuilder = InvalidateTokensResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Activate an app for a user + * + * Activates a user for a given oauth client app. This indicates the user has been informed that + * the app will have access to the user's looker data, and that the user has accepted and allowed + * the app to use their Looker account. + * + * Activating a user for an app that the user is already activated with returns a success response. + * + */ + @Override + public void activateAppUser(ActivateAppUserReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/oauth_client_apps/{client_guid}/users/{user_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ActivateAppUserResponse.Builder responseBuilder = ActivateAppUserResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Deactivate an app for a user + * + * Deactivate a user for a given oauth client app. All tokens issued to the app for + * this user will be invalid immediately. Before the user can use the app with their + * Looker account, the user will have to read and accept an account use disclosure statement for the app. + * + * Admin users can deactivate other users, but non-admin users can only deactivate themselves. + * + * As with most REST DELETE operations, this endpoint does not return an error if the indicated + * resource (app or user) does not exist or has already been deactivated. + * + */ + @Override + public void deactivateAppUser(DeactivateAppUserReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/oauth_client_apps/{client_guid}/users/{user_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeactivateAppUserResponse.Builder responseBuilder = DeactivateAppUserResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get the OIDC configuration. + * + * Looker can be optionally configured to authenticate users against an OpenID Connect (OIDC) + * authentication server. OIDC setup requires coordination with an administrator of that server. + * + * Only Looker administrators can read and update the OIDC configuration. + * + * Configuring OIDC impacts authentication for all users. This configuration should be done carefully. + * + * Looker maintains a single OIDC configuation. It can be read and updated. Updates only succeed if the new state will be valid (in the sense that all required fields are populated); it is up to you to ensure that the configuration is appropriate and correct). + * + * OIDC is enabled or disabled for Looker using the **enabled** field. + * + */ + @Override + public void oidcConfig(OidcConfigReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/oidc_config", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + OidcConfigResponse.Builder responseBuilder = OidcConfigResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update the OIDC configuration. + * + * Configuring OIDC impacts authentication for all users. This configuration should be done carefully. + * + * Only Looker administrators can read and update the OIDC configuration. + * + * OIDC is enabled or disabled for Looker using the **enabled** field. + * + * It is **highly** recommended that any OIDC setting changes be tested using the APIs below before being set globally. + * + */ + @Override + public void updateOidcConfig(UpdateOidcConfigReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/oidc_config", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateOidcConfigResponse.Builder responseBuilder = UpdateOidcConfigResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get a OIDC test configuration by test_slug. + * + */ + @Override + public void oidcTestConfig(OidcTestConfigReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/oidc_test_configs/{test_slug}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + OidcTestConfigResponse.Builder responseBuilder = OidcTestConfigResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Delete a OIDC test configuration. + * + */ + @Override + public void deleteOidcTestConfig(DeleteOidcTestConfigReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/oidc_test_configs/{test_slug}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteOidcTestConfigResponse.Builder responseBuilder = DeleteOidcTestConfigResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create a OIDC test configuration. + * + */ + @Override + public void createOidcTestConfig(CreateOidcTestConfigReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/oidc_test_configs", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateOidcTestConfigResponse.Builder responseBuilder = CreateOidcTestConfigResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get password config. + * + */ + @Override + public void passwordConfig(PasswordConfigReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/password_config", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + PasswordConfigResponse.Builder responseBuilder = PasswordConfigResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update password config. + * + */ + @Override + public void updatePasswordConfig(UpdatePasswordConfigReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/password_config", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdatePasswordConfigResponse.Builder responseBuilder = UpdatePasswordConfigResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Force all credentials_email users to reset their login passwords upon their next login. + * + */ + @Override + public void forcePasswordResetAtNextLoginForAllUsers(ForcePasswordResetAtNextLoginForAllUsersReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.put("/password_config/force_password_reset_at_next_login_for_all_users", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ForcePasswordResetAtNextLoginForAllUsersResponse.Builder responseBuilder = ForcePasswordResetAtNextLoginForAllUsersResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get the SAML configuration. + * + * Looker can be optionally configured to authenticate users against a SAML authentication server. + * SAML setup requires coordination with an administrator of that server. + * + * Only Looker administrators can read and update the SAML configuration. + * + * Configuring SAML impacts authentication for all users. This configuration should be done carefully. + * + * Looker maintains a single SAML configuation. It can be read and updated. Updates only succeed if the new state will be valid (in the sense that all required fields are populated); it is up to you to ensure that the configuration is appropriate and correct). + * + * SAML is enabled or disabled for Looker using the **enabled** field. + * + */ + @Override + public void samlConfig(SamlConfigReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/saml_config", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SamlConfigResponse.Builder responseBuilder = SamlConfigResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update the SAML configuration. + * + * Configuring SAML impacts authentication for all users. This configuration should be done carefully. + * + * Only Looker administrators can read and update the SAML configuration. + * + * SAML is enabled or disabled for Looker using the **enabled** field. + * + * It is **highly** recommended that any SAML setting changes be tested using the APIs below before being set globally. + * + */ + @Override + public void updateSamlConfig(UpdateSamlConfigReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/saml_config", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateSamlConfigResponse.Builder responseBuilder = UpdateSamlConfigResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get a SAML test configuration by test_slug. + * + */ + @Override + public void samlTestConfig(SamlTestConfigReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/saml_test_configs/{test_slug}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SamlTestConfigResponse.Builder responseBuilder = SamlTestConfigResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Delete a SAML test configuration. + * + */ + @Override + public void deleteSamlTestConfig(DeleteSamlTestConfigReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/saml_test_configs/{test_slug}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteSamlTestConfigResponse.Builder responseBuilder = DeleteSamlTestConfigResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create a SAML test configuration. + * + */ + @Override + public void createSamlTestConfig(CreateSamlTestConfigReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/saml_test_configs", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateSamlTestConfigResponse.Builder responseBuilder = CreateSamlTestConfigResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Parse the given xml as a SAML IdP metadata document and return the result. + * + */ + @Override + public void parseSamlIdpMetadata(ParseSamlIdpMetadataReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/parse_saml_idp_metadata", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ParseSamlIdpMetadataResponse.Builder responseBuilder = ParseSamlIdpMetadataResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Fetch the given url and parse it as a SAML IdP metadata document and return the result. + * Note that this requires that the url be public or at least at a location where the Looker instance + * can fetch it without requiring any special authentication. + * + */ + @Override + public void fetchAndParseSamlIdpMetadata(FetchAndParseSamlIdpMetadataReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/fetch_and_parse_saml_idp_metadata", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + FetchAndParseSamlIdpMetadataResponse.Builder responseBuilder = FetchAndParseSamlIdpMetadataResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get session config. + * + */ + @Override + public void sessionConfig(SessionConfigReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/session_config", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SessionConfigResponse.Builder responseBuilder = SessionConfigResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update session config. + * + */ + @Override + public void updateSessionConfig(UpdateSessionConfigReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/session_config", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateSessionConfigResponse.Builder responseBuilder = UpdateSessionConfigResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get Support Access Allowlist Users + * + * Returns the users that have been added to the Support Access Allowlist + * + */ + @Override + public void getSupportAccessAllowlistEntries(GetSupportAccessAllowlistEntriesReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/support_access/allowlist", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + GetSupportAccessAllowlistEntriesResponse.Builder responseBuilder = GetSupportAccessAllowlistEntriesResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Add Support Access Allowlist Users + * + * Adds a list of emails to the Allowlist, using the provided reason + * + */ + @Override + public void addSupportAccessAllowlistEntries(AddSupportAccessAllowlistEntriesReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/support_access/allowlist", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AddSupportAccessAllowlistEntriesResponse.Builder responseBuilder = AddSupportAccessAllowlistEntriesResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Delete Support Access Allowlist User + * + * Deletes the specified Allowlist Entry Id + * + */ + @Override + public void deleteSupportAccessAllowlistEntry(DeleteSupportAccessAllowlistEntryReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/support_access/allowlist/{entry_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteSupportAccessAllowlistEntryResponse.Builder responseBuilder = DeleteSupportAccessAllowlistEntryResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Enable Support Access + * + * Enables Support Access for the provided duration + * + */ + @Override + public void enableSupportAccess(EnableSupportAccessReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.put("/support_access/enable", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + EnableSupportAccessResponse.Builder responseBuilder = EnableSupportAccessResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Disable Support Access + * + * Disables Support Access immediately + * + */ + @Override + public void disableSupportAccess(DisableSupportAccessReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.put("/support_access/disable", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DisableSupportAccessResponse.Builder responseBuilder = DisableSupportAccessResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Support Access Status + * + * Returns the current Support Access Status + * + */ + @Override + public void supportAccessStatus(SupportAccessStatusReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/support_access/status", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SupportAccessStatusResponse.Builder responseBuilder = SupportAccessStatusResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get currently locked-out users. + * + */ + @Override + public void allUserLoginLockouts(AllUserLoginLockoutsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/user_login_lockouts", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllUserLoginLockoutsResponse.Builder responseBuilder = AllUserLoginLockoutsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Search currently locked-out users. + * + */ + @Override + public void searchUserLoginLockouts(SearchUserLoginLockoutsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/user_login_lockouts/search", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SearchUserLoginLockoutsResponse.Builder responseBuilder = SearchUserLoginLockoutsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Removes login lockout for the associated user. + * + */ + @Override + public void deleteUserLoginLockout(DeleteUserLoginLockoutReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/user_login_lockout/{key}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteUserLoginLockoutResponse.Builder responseBuilder = DeleteUserLoginLockoutResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + //#endregion Auth: Manage User Authentication Configuration + + //#region Board: Manage Boards + + /** + * ### Get information about all boards. + * + */ + @Override + public void allBoards(AllBoardsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/boards", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllBoardsResponse.Builder responseBuilder = AllBoardsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create a new board. + * + */ + @Override + public void createBoard(CreateBoardReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/boards", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateBoardResponse.Builder responseBuilder = CreateBoardResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Search Boards + * + * If multiple search params are given and `filter_or` is FALSE or not specified, + * search params are combined in a logical AND operation. + * Only rows that match *all* search param criteria will be returned. + * + * If `filter_or` is TRUE, multiple search params are combined in a logical OR operation. + * Results will include rows that match **any** of the search criteria. + * + * String search params use case-insensitive matching. + * String search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions. + * example="dan%" will match "danger" and "Danzig" but not "David" + * example="D_m%" will match "Damage" and "dump" + * + * Integer search params can accept a single value or a comma separated list of values. The multiple + * values will be combined under a logical OR operation - results will match at least one of + * the given values. + * + * Most search params can accept "IS NULL" and "NOT NULL" as special expressions to match + * or exclude (respectively) rows where the column is null. + * + * Boolean search params accept only "true" and "false" as values. + * + * + */ + @Override + public void searchBoards(SearchBoardsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/boards/search", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SearchBoardsResponse.Builder responseBuilder = SearchBoardsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about a board. + * + */ + @Override + public void board(BoardReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/boards/{board_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + BoardResponse.Builder responseBuilder = BoardResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update a board definition. + * + */ + @Override + public void updateBoard(UpdateBoardReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/boards/{board_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateBoardResponse.Builder responseBuilder = UpdateBoardResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Delete a board. + * + */ + @Override + public void deleteBoard(DeleteBoardReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/boards/{board_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteBoardResponse.Builder responseBuilder = DeleteBoardResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about all board items. + * + */ + @Override + public void allBoardItems(AllBoardItemsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/board_items", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllBoardItemsResponse.Builder responseBuilder = AllBoardItemsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create a new board item. + * + */ + @Override + public void createBoardItem(CreateBoardItemReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/board_items", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateBoardItemResponse.Builder responseBuilder = CreateBoardItemResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about a board item. + * + */ + @Override + public void boardItem(BoardItemReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/board_items/{board_item_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + BoardItemResponse.Builder responseBuilder = BoardItemResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update a board item definition. + * + */ + @Override + public void updateBoardItem(UpdateBoardItemReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/board_items/{board_item_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateBoardItemResponse.Builder responseBuilder = UpdateBoardItemResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Delete a board item. + * + */ + @Override + public void deleteBoardItem(DeleteBoardItemReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/board_items/{board_item_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteBoardItemResponse.Builder responseBuilder = DeleteBoardItemResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about all board sections. + * + */ + @Override + public void allBoardSections(AllBoardSectionsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/board_sections", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllBoardSectionsResponse.Builder responseBuilder = AllBoardSectionsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create a new board section. + * + */ + @Override + public void createBoardSection(CreateBoardSectionReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/board_sections", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateBoardSectionResponse.Builder responseBuilder = CreateBoardSectionResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about a board section. + * + */ + @Override + public void boardSection(BoardSectionReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/board_sections/{board_section_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + BoardSectionResponse.Builder responseBuilder = BoardSectionResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update a board section definition. + * + */ + @Override + public void updateBoardSection(UpdateBoardSectionReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/board_sections/{board_section_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateBoardSectionResponse.Builder responseBuilder = UpdateBoardSectionResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Delete a board section. + * + */ + @Override + public void deleteBoardSection(DeleteBoardSectionReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/board_sections/{board_section_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteBoardSectionResponse.Builder responseBuilder = DeleteBoardSectionResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + //#endregion Board: Manage Boards + + //#region ColorCollection: Manage Color Collections + + /** + * ### Get an array of all existing Color Collections + * Get a **single** color collection by id with [ColorCollection](#!/ColorCollection/color_collection) + * + * Get all **standard** color collections with [ColorCollection](#!/ColorCollection/color_collections_standard) + * + * Get all **custom** color collections with [ColorCollection](#!/ColorCollection/color_collections_custom) + * + * **Note**: Only an API user with the Admin role can call this endpoint. Unauthorized requests will return `Not Found` (404) errors. + * + * + */ + @Override + public void allColorCollections(AllColorCollectionsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/color_collections", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllColorCollectionsResponse.Builder responseBuilder = AllColorCollectionsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create a custom color collection with the specified information + * + * Creates a new custom color collection object, returning the details, including the created id. + * + * **Update** an existing color collection with [Update Color Collection](#!/ColorCollection/update_color_collection) + * + * **Permanently delete** an existing custom color collection with [Delete Color Collection](#!/ColorCollection/delete_color_collection) + * + * **Note**: Only an API user with the Admin role can call this endpoint. Unauthorized requests will return `Not Found` (404) errors. + * + * + */ + @Override + public void createColorCollection(CreateColorCollectionReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/color_collections", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateColorCollectionResponse.Builder responseBuilder = CreateColorCollectionResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get an array of all existing **Custom** Color Collections + * Get a **single** color collection by id with [ColorCollection](#!/ColorCollection/color_collection) + * + * Get all **standard** color collections with [ColorCollection](#!/ColorCollection/color_collections_standard) + * + * **Note**: Only an API user with the Admin role can call this endpoint. Unauthorized requests will return `Not Found` (404) errors. + * + * + */ + @Override + public void colorCollectionsCustom(ColorCollectionsCustomReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/color_collections/custom", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ColorCollectionsCustomResponse.Builder responseBuilder = ColorCollectionsCustomResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get an array of all existing **Standard** Color Collections + * Get a **single** color collection by id with [ColorCollection](#!/ColorCollection/color_collection) + * + * Get all **custom** color collections with [ColorCollection](#!/ColorCollection/color_collections_custom) + * + * **Note**: Only an API user with the Admin role can call this endpoint. Unauthorized requests will return `Not Found` (404) errors. + * + * + */ + @Override + public void colorCollectionsStandard(ColorCollectionsStandardReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/color_collections/standard", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ColorCollectionsStandardResponse.Builder responseBuilder = ColorCollectionsStandardResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get the default color collection + * + * Use this to retrieve the default Color Collection. + * + * Set the default color collection with [ColorCollection](#!/ColorCollection/set_default_color_collection) + * + */ + @Override + public void defaultColorCollection(DefaultColorCollectionReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/color_collections/default", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DefaultColorCollectionResponse.Builder responseBuilder = DefaultColorCollectionResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Set the global default Color Collection by ID + * + * Returns the new specified default Color Collection object. + * **Note**: Only an API user with the Admin role can call this endpoint. Unauthorized requests will return `Not Found` (404) errors. + * + * + */ + @Override + public void setDefaultColorCollection(SetDefaultColorCollectionReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.put("/color_collections/default", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SetDefaultColorCollectionResponse.Builder responseBuilder = SetDefaultColorCollectionResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get a Color Collection by ID + * + * Use this to retrieve a specific Color Collection. + * Get a **single** color collection by id with [ColorCollection](#!/ColorCollection/color_collection) + * + * Get all **standard** color collections with [ColorCollection](#!/ColorCollection/color_collections_standard) + * + * Get all **custom** color collections with [ColorCollection](#!/ColorCollection/color_collections_custom) + * + * **Note**: Only an API user with the Admin role can call this endpoint. Unauthorized requests will return `Not Found` (404) errors. + * + * + */ + @Override + public void colorCollection(ColorCollectionReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/color_collections/{collection_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ColorCollectionResponse.Builder responseBuilder = ColorCollectionResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update a custom color collection by id. + * **Note**: Only an API user with the Admin role can call this endpoint. Unauthorized requests will return `Not Found` (404) errors. + * + * + */ + @Override + public void updateColorCollection(UpdateColorCollectionReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/color_collections/{collection_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateColorCollectionResponse.Builder responseBuilder = UpdateColorCollectionResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Delete a custom color collection by id + * + * This operation permanently deletes the identified **Custom** color collection. + * + * **Standard** color collections cannot be deleted + * + * Because multiple color collections can have the same label, they must be deleted by ID, not name. + * **Note**: Only an API user with the Admin role can call this endpoint. Unauthorized requests will return `Not Found` (404) errors. + * + * + */ + @Override + public void deleteColorCollection(DeleteColorCollectionReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/color_collections/{collection_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteColorCollectionResponse.Builder responseBuilder = DeleteColorCollectionResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + //#endregion ColorCollection: Manage Color Collections + + //#region Config: Manage General Configuration + + /** + * Get the current Cloud Storage Configuration. + * + */ + @Override + public void cloudStorageConfiguration(CloudStorageConfigurationReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/cloud_storage", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CloudStorageConfigurationResponse.Builder responseBuilder = CloudStorageConfigurationResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * Update the current Cloud Storage Configuration. + * + */ + @Override + public void updateCloudStorageConfiguration(UpdateCloudStorageConfigurationReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/cloud_storage", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateCloudStorageConfigurationResponse.Builder responseBuilder = UpdateCloudStorageConfigurationResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get the current status and content of custom welcome emails + * + */ + @Override + public void customWelcomeEmail(CustomWelcomeEmailReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/custom_welcome_email", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CustomWelcomeEmailResponse.Builder responseBuilder = CustomWelcomeEmailResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * Update custom welcome email setting and values. Optionally send a test email with the new content to the currently logged in user. + * + */ + @Override + public void updateCustomWelcomeEmail(UpdateCustomWelcomeEmailReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/custom_welcome_email", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateCustomWelcomeEmailResponse.Builder responseBuilder = UpdateCustomWelcomeEmailResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * Requests to this endpoint will send a welcome email with the custom content provided in the body to the currently logged in user. + * + */ + @Override + public void updateCustomWelcomeEmailTest(UpdateCustomWelcomeEmailTestReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.put("/custom_welcome_email_test", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateCustomWelcomeEmailTestResponse.Builder responseBuilder = UpdateCustomWelcomeEmailTestResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Retrieve the value for whether or not digest emails is enabled + * + */ + @Override + public void digestEmailsEnabled(DigestEmailsEnabledReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/digest_emails_enabled", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DigestEmailsEnabledResponse.Builder responseBuilder = DigestEmailsEnabledResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update the setting for enabling/disabling digest emails + * + */ + @Override + public void updateDigestEmailsEnabled(UpdateDigestEmailsEnabledReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/digest_emails_enabled", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateDigestEmailsEnabledResponse.Builder responseBuilder = UpdateDigestEmailsEnabledResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Trigger the generation of digest email records and send them to Looker's internal system. This does not send + * any actual emails, it generates records containing content which may be of interest for users who have become inactive. + * Emails will be sent at a later time from Looker's internal system if the Digest Emails feature is enabled in settings. + */ + @Override + public void createDigestEmailSend(CreateDigestEmailSendReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/digest_email_send", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateDigestEmailSendResponse.Builder responseBuilder = CreateDigestEmailSendResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get Egress IP Addresses + * + * Returns the list of public egress IP Addresses for a hosted customer's instance + * + */ + @Override + public void publicEgressIpAddresses(PublicEgressIpAddressesReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/public_egress_ip_addresses", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + PublicEgressIpAddressesResponse.Builder responseBuilder = PublicEgressIpAddressesResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Set the menu item name and content for internal help resources + * + */ + @Override + public void internalHelpResourcesContent(InternalHelpResourcesContentReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/internal_help_resources_content", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + InternalHelpResourcesContentResponse.Builder responseBuilder = InternalHelpResourcesContentResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * Update internal help resources content + * + */ + @Override + public void updateInternalHelpResourcesContent(UpdateInternalHelpResourcesContentReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/internal_help_resources_content", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateInternalHelpResourcesContentResponse.Builder responseBuilder = UpdateInternalHelpResourcesContentResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get and set the options for internal help resources + * + */ + @Override + public void internalHelpResources(InternalHelpResourcesReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/internal_help_resources_enabled", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + InternalHelpResourcesResponse.Builder responseBuilder = InternalHelpResourcesResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * Update internal help resources settings + * + */ + @Override + public void updateInternalHelpResources(UpdateInternalHelpResourcesReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/internal_help_resources", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateInternalHelpResourcesResponse.Builder responseBuilder = UpdateInternalHelpResourcesResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get all legacy features. + * + */ + @Override + public void allLegacyFeatures(AllLegacyFeaturesReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/legacy_features", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllLegacyFeaturesResponse.Builder responseBuilder = AllLegacyFeaturesResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about the legacy feature with a specific id. + * + */ + @Override + public void legacyFeature(LegacyFeatureReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/legacy_features/{legacy_feature_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + LegacyFeatureResponse.Builder responseBuilder = LegacyFeatureResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update information about the legacy feature with a specific id. + * + */ + @Override + public void updateLegacyFeature(UpdateLegacyFeatureReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/legacy_features/{legacy_feature_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateLegacyFeatureResponse.Builder responseBuilder = UpdateLegacyFeatureResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get a list of locales that Looker supports. + * + */ + @Override + public void allLocales(AllLocalesReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/locales", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllLocalesResponse.Builder responseBuilder = AllLocalesResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get all mobile settings. + * + */ + @Override + public void mobileSettings(MobileSettingsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/mobile/settings", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + MobileSettingsResponse.Builder responseBuilder = MobileSettingsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get Looker Settings + * + * Available settings are: + * - extension_framework_enabled + * - marketplace_auto_install_enabled + * - marketplace_enabled + * - privatelabel_configuration + * - custom_welcome_email + * - onboarding_enabled + * + * + */ + @Override + public void getSetting(GetSettingReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/setting", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + GetSettingResponse.Builder responseBuilder = GetSettingResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Configure Looker Settings + * + * Available settings are: + * - extension_framework_enabled + * - marketplace_auto_install_enabled + * - marketplace_enabled + * - privatelabel_configuration + * - custom_welcome_email + * - onboarding_enabled + * + * See the `Setting` type for more information on the specific values that can be configured. + * + */ + @Override + public void setSetting(SetSettingReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/setting", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SetSettingResponse.Builder responseBuilder = SetSettingResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Configure SMTP Settings + * This API allows users to configure the SMTP settings on the Looker instance. + * This API is only supported in the OEM jar. Additionally, only admin users are authorised to call this API. + * + */ + @Override + public void setSmtpSettings(SetSmtpSettingsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/smtp_settings", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SetSmtpSettingsResponse.Builder responseBuilder = SetSmtpSettingsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get current SMTP status. + * + */ + @Override + public void smtpStatus(SmtpStatusReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/smtp_status", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SmtpStatusResponse.Builder responseBuilder = SmtpStatusResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get a list of timezones that Looker supports (e.g. useful for scheduling tasks). + * + */ + @Override + public void allTimezones(AllTimezonesReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/timezones", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllTimezonesResponse.Builder responseBuilder = AllTimezonesResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about all API versions supported by this Looker instance. + * + */ + @Override + public void versions(VersionsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/versions", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + VersionsResponse.Builder responseBuilder = VersionsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get an API specification for this Looker instance. + * + * The specification is returned as a JSON document in Swagger 2.x format + * + */ + @Override + public void apiSpec(ApiSpecReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/api_spec/{api_version}/{specification}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ApiSpecResponse.Builder responseBuilder = ApiSpecResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### This feature is enabled only by special license. + * ### Gets the whitelabel configuration, which includes hiding documentation links, custom favicon uploading, etc. + * + */ + @Override + public void whitelabelConfiguration(WhitelabelConfigurationReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/whitelabel_configuration", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + WhitelabelConfigurationResponse.Builder responseBuilder = WhitelabelConfigurationResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update the whitelabel configuration + * + */ + @Override + public void updateWhitelabelConfiguration(UpdateWhitelabelConfigurationReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.put("/whitelabel_configuration", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateWhitelabelConfigurationResponse.Builder responseBuilder = UpdateWhitelabelConfigurationResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + //#endregion Config: Manage General Configuration + + //#region Connection: Manage Database Connections + + /** + * ### Get information about all connections. + * + */ + @Override + public void allConnections(AllConnectionsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/connections", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllConnectionsResponse.Builder responseBuilder = AllConnectionsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create a connection using the specified configuration. + * + */ + @Override + public void createConnection(CreateConnectionReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/connections", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateConnectionResponse.Builder responseBuilder = CreateConnectionResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about a connection. + * + */ + @Override + public void connection(ConnectionReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/connections/{connection_name}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ConnectionResponse.Builder responseBuilder = ConnectionResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update a connection using the specified configuration. + * + */ + @Override + public void updateConnection(UpdateConnectionReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/connections/{connection_name}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateConnectionResponse.Builder responseBuilder = UpdateConnectionResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Delete a connection. + * + */ + @Override + public void deleteConnection(DeleteConnectionReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/connections/{connection_name}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteConnectionResponse.Builder responseBuilder = DeleteConnectionResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Delete a connection override. + * + */ + @Override + public void deleteConnectionOverride(DeleteConnectionOverrideReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/connections/{connection_name}/connection_override/{override_context}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteConnectionOverrideResponse.Builder responseBuilder = DeleteConnectionOverrideResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Test an existing connection. + * + * Note that a connection's 'dialect' property has a 'connection_tests' property that lists the + * specific types of tests that the connection supports. + * + * This API is rate limited. + * + * Unsupported tests in the request will be ignored. + * + */ + @Override + public void testConnection(TestConnectionReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.put("/connections/{connection_name}/test", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + TestConnectionResponse.Builder responseBuilder = TestConnectionResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Test a connection configuration. + * + * Note that a connection's 'dialect' property has a 'connection_tests' property that lists the + * specific types of tests that the connection supports. + * + * This API is rate limited. + * + * Unsupported tests in the request will be ignored. + * + */ + @Override + public void testConnectionConfig(TestConnectionConfigReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.put("/connections/test", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + TestConnectionConfigResponse.Builder responseBuilder = TestConnectionConfigResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about all dialects. + * + */ + @Override + public void allDialectInfos(AllDialectInfosReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/dialect_info", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllDialectInfosResponse.Builder responseBuilder = AllDialectInfosResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get all External OAuth Applications. + * + * This is an OAuth Application which Looker uses to access external systems. + * + */ + @Override + public void allExternalOauthApplications(AllExternalOauthApplicationsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/external_oauth_applications", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllExternalOauthApplicationsResponse.Builder responseBuilder = AllExternalOauthApplicationsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create an OAuth Application using the specified configuration. + * + * This is an OAuth Application which Looker uses to access external systems. + * + */ + @Override + public void createExternalOauthApplication(CreateExternalOauthApplicationReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/external_oauth_applications", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateExternalOauthApplicationResponse.Builder responseBuilder = CreateExternalOauthApplicationResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create OAuth User state. + * + */ + @Override + public void createOauthApplicationUserState(CreateOauthApplicationUserStateReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/external_oauth_applications/user_state", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateOauthApplicationUserStateResponse.Builder responseBuilder = CreateOauthApplicationUserStateResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about all SSH Servers. + * + */ + @Override + public void allSshServers(AllSshServersReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/ssh_servers", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllSshServersResponse.Builder responseBuilder = AllSshServersResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create an SSH Server. + * + */ + @Override + public void createSshServer(CreateSshServerReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/ssh_servers", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateSshServerResponse.Builder responseBuilder = CreateSshServerResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about an SSH Server. + * + */ + @Override + public void sshServer(SshServerReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/ssh_server/{ssh_server_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SshServerResponse.Builder responseBuilder = SshServerResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update an SSH Server. + * + */ + @Override + public void updateSshServer(UpdateSshServerReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/ssh_server/{ssh_server_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateSshServerResponse.Builder responseBuilder = UpdateSshServerResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Delete an SSH Server. + * + */ + @Override + public void deleteSshServer(DeleteSshServerReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/ssh_server/{ssh_server_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteSshServerResponse.Builder responseBuilder = DeleteSshServerResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Test the SSH Server + * + */ + @Override + public void testSshServer(TestSshServerReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/ssh_server/{ssh_server_id}/test", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + TestSshServerResponse.Builder responseBuilder = TestSshServerResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about all SSH Tunnels. + * + */ + @Override + public void allSshTunnels(AllSshTunnelsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/ssh_tunnels", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllSshTunnelsResponse.Builder responseBuilder = AllSshTunnelsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create an SSH Tunnel + * + */ + @Override + public void createSshTunnel(CreateSshTunnelReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/ssh_tunnels", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateSshTunnelResponse.Builder responseBuilder = CreateSshTunnelResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about an SSH Tunnel. + * + */ + @Override + public void sshTunnel(SshTunnelReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/ssh_tunnel/{ssh_tunnel_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SshTunnelResponse.Builder responseBuilder = SshTunnelResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update an SSH Tunnel + * + */ + @Override + public void updateSshTunnel(UpdateSshTunnelReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/ssh_tunnel/{ssh_tunnel_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateSshTunnelResponse.Builder responseBuilder = UpdateSshTunnelResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Delete an SSH Tunnel + * + */ + @Override + public void deleteSshTunnel(DeleteSshTunnelReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/ssh_tunnel/{ssh_tunnel_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteSshTunnelResponse.Builder responseBuilder = DeleteSshTunnelResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Test the SSH Tunnel + * + */ + @Override + public void testSshTunnel(TestSshTunnelReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/ssh_tunnel/{ssh_tunnel_id}/test", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + TestSshTunnelResponse.Builder responseBuilder = TestSshTunnelResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get the SSH public key + * + * Get the public key created for this instance to identify itself to a remote SSH server. + * + */ + @Override + public void sshPublicKey(SshPublicKeyReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/ssh_public_key", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SshPublicKeyResponse.Builder responseBuilder = SshPublicKeyResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + //#endregion Connection: Manage Database Connections + + //#region Content: Manage Content + + /** + * ### Search Favorite Content + * + * If multiple search params are given and `filter_or` is FALSE or not specified, + * search params are combined in a logical AND operation. + * Only rows that match *all* search param criteria will be returned. + * + * If `filter_or` is TRUE, multiple search params are combined in a logical OR operation. + * Results will include rows that match **any** of the search criteria. + * + * String search params use case-insensitive matching. + * String search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions. + * example="dan%" will match "danger" and "Danzig" but not "David" + * example="D_m%" will match "Damage" and "dump" + * + * Integer search params can accept a single value or a comma separated list of values. The multiple + * values will be combined under a logical OR operation - results will match at least one of + * the given values. + * + * Most search params can accept "IS NULL" and "NOT NULL" as special expressions to match + * or exclude (respectively) rows where the column is null. + * + * Boolean search params accept only "true" and "false" as values. + * + * + */ + @Override + public void searchContentFavorites(SearchContentFavoritesReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/content_favorite/search", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SearchContentFavoritesResponse.Builder responseBuilder = SearchContentFavoritesResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get favorite content by its id + */ + @Override + public void contentFavorite(ContentFavoriteReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/content_favorite/{content_favorite_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ContentFavoriteResponse.Builder responseBuilder = ContentFavoriteResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Delete favorite content + */ + @Override + public void deleteContentFavorite(DeleteContentFavoriteReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/content_favorite/{content_favorite_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteContentFavoriteResponse.Builder responseBuilder = DeleteContentFavoriteResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create favorite content + */ + @Override + public void createContentFavorite(CreateContentFavoriteReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/content_favorite", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateContentFavoriteResponse.Builder responseBuilder = CreateContentFavoriteResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about all content metadata in a space. + * + */ + @Override + public void allContentMetadatas(AllContentMetadatasReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/content_metadata", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllContentMetadatasResponse.Builder responseBuilder = AllContentMetadatasResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about an individual content metadata record. + * + */ + @Override + public void contentMetadata(ContentMetadataReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/content_metadata/{content_metadata_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ContentMetadataResponse.Builder responseBuilder = ContentMetadataResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Move a piece of content. + * + */ + @Override + public void updateContentMetadata(UpdateContentMetadataReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/content_metadata/{content_metadata_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateContentMetadataResponse.Builder responseBuilder = UpdateContentMetadataResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### All content metadata access records for a content metadata item. + * + */ + @Override + public void allContentMetadataAccesses(AllContentMetadataAccessesReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/content_metadata_access", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllContentMetadataAccessesResponse.Builder responseBuilder = AllContentMetadataAccessesResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create content metadata access. + * + */ + @Override + public void createContentMetadataAccess(CreateContentMetadataAccessReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/content_metadata_access", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateContentMetadataAccessResponse.Builder responseBuilder = CreateContentMetadataAccessResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update type of access for content metadata. + * + */ + @Override + public void updateContentMetadataAccess(UpdateContentMetadataAccessReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.put("/content_metadata_access/{content_metadata_access_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateContentMetadataAccessResponse.Builder responseBuilder = UpdateContentMetadataAccessResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Remove content metadata access. + * + */ + @Override + public void deleteContentMetadataAccess(DeleteContentMetadataAccessReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/content_metadata_access/{content_metadata_access_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteContentMetadataAccessResponse.Builder responseBuilder = DeleteContentMetadataAccessResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get an image representing the contents of a dashboard or look. + * + * The returned thumbnail is an abstract representation of the contents of a dashbord or look and does not + * reflect the actual data displayed in the respective visualizations. + * + */ + @Override + public void contentThumbnail(ContentThumbnailReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/content_thumbnail/{type}/{resource_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ContentThumbnailResponse.Builder responseBuilder = ContentThumbnailResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Validate All Content + * + * Performs validation of all looks and dashboards + * Returns a list of errors found as well as metadata about the content validation run. + * + */ + @Override + public void contentValidation(ContentValidationReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/content_validation", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ContentValidationResponse.Builder responseBuilder = ContentValidationResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Search Content Views + * + * If multiple search params are given and `filter_or` is FALSE or not specified, + * search params are combined in a logical AND operation. + * Only rows that match *all* search param criteria will be returned. + * + * If `filter_or` is TRUE, multiple search params are combined in a logical OR operation. + * Results will include rows that match **any** of the search criteria. + * + * String search params use case-insensitive matching. + * String search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions. + * example="dan%" will match "danger" and "Danzig" but not "David" + * example="D_m%" will match "Damage" and "dump" + * + * Integer search params can accept a single value or a comma separated list of values. The multiple + * values will be combined under a logical OR operation - results will match at least one of + * the given values. + * + * Most search params can accept "IS NULL" and "NOT NULL" as special expressions to match + * or exclude (respectively) rows where the column is null. + * + * Boolean search params accept only "true" and "false" as values. + * + * + */ + @Override + public void searchContentViews(SearchContentViewsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/content_view/search", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SearchContentViewsResponse.Builder responseBuilder = SearchContentViewsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get a vector image representing the contents of a dashboard or look. + * + * # DEPRECATED: Use [content_thumbnail()](#!/Content/content_thumbnail) + * + * The returned thumbnail is an abstract representation of the contents of a dashbord or look and does not + * reflect the actual data displayed in the respective visualizations. + * + */ + @Override + public void vectorThumbnail(VectorThumbnailReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/vector_thumbnail/{type}/{resource_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + VectorThumbnailResponse.Builder responseBuilder = VectorThumbnailResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + //#endregion Content: Manage Content + + //#region Dashboard: Manage Dashboards + + /** + * ### Get information about all active dashboards. + * + * Returns an array of **abbreviated dashboard objects**. Dashboards marked as deleted are excluded from this list. + * + * Get the **full details** of a specific dashboard by id with [dashboard()](#!/Dashboard/dashboard) + * + * Find **deleted dashboards** with [search_dashboards()](#!/Dashboard/search_dashboards) + * + */ + @Override + public void allDashboards(AllDashboardsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/dashboards", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllDashboardsResponse.Builder responseBuilder = AllDashboardsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create a new dashboard + * + * Creates a new dashboard object and returns the details of the newly created dashboard. + * + * `Title` and `space_id` are required fields. + * `Space_id` must contain the id of an existing space. + * A dashboard's `title` must be unique within the space in which it resides. + * + * If you receive a 422 error response when creating a dashboard, be sure to look at the + * response body for information about exactly which fields are missing or contain invalid data. + * + * You can **update** an existing dashboard with [update_dashboard()](#!/Dashboard/update_dashboard) + * + * You can **permanently delete** an existing dashboard with [delete_dashboard()](#!/Dashboard/delete_dashboard) + * + */ + @Override + public void createDashboard(CreateDashboardReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/dashboards", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateDashboardResponse.Builder responseBuilder = CreateDashboardResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Search Dashboards + * + * Returns an **array of dashboard objects** that match the specified search criteria. + * + * If multiple search params are given and `filter_or` is FALSE or not specified, + * search params are combined in a logical AND operation. + * Only rows that match *all* search param criteria will be returned. + * + * If `filter_or` is TRUE, multiple search params are combined in a logical OR operation. + * Results will include rows that match **any** of the search criteria. + * + * String search params use case-insensitive matching. + * String search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions. + * example="dan%" will match "danger" and "Danzig" but not "David" + * example="D_m%" will match "Damage" and "dump" + * + * Integer search params can accept a single value or a comma separated list of values. The multiple + * values will be combined under a logical OR operation - results will match at least one of + * the given values. + * + * Most search params can accept "IS NULL" and "NOT NULL" as special expressions to match + * or exclude (respectively) rows where the column is null. + * + * Boolean search params accept only "true" and "false" as values. + * + * + * The parameters `limit`, and `offset` are recommended for fetching results in page-size chunks. + * + * Get a **single dashboard** by id with [dashboard()](#!/Dashboard/dashboard) + * + */ + @Override + public void searchDashboards(SearchDashboardsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/dashboards/search", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SearchDashboardsResponse.Builder responseBuilder = SearchDashboardsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Import a LookML dashboard to a space as a UDD + * Creates a UDD (a dashboard which exists in the Looker database rather than as a LookML file) from the LookML dashboard + * and places it in the space specified. The created UDD will have a lookml_link_id which links to the original LookML dashboard. + * + * To give the imported dashboard specify a (e.g. title: "my title") in the body of your request, otherwise the imported + * dashboard will have the same title as the original LookML dashboard. + * + * For this operation to succeed the user must have permission to see the LookML dashboard in question, and have permission to + * create content in the space the dashboard is being imported to. + * + * **Sync** a linked UDD with [sync_lookml_dashboard()](#!/Dashboard/sync_lookml_dashboard) + * **Unlink** a linked UDD by setting lookml_link_id to null with [update_dashboard()](#!/Dashboard/update_dashboard) + * + */ + @Override + public void importLookmlDashboard(ImportLookmlDashboardReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/dashboards/{lookml_dashboard_id}/import/{space_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ImportLookmlDashboardResponse.Builder responseBuilder = ImportLookmlDashboardResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update all linked dashboards to match the specified LookML dashboard. + * + * Any UDD (a dashboard which exists in the Looker database rather than as a LookML file) which has a `lookml_link_id` + * property value referring to a LookML dashboard's id (model::dashboardname) will be updated so that it matches the current state of the LookML dashboard. + * + * For this operation to succeed the user must have permission to view the LookML dashboard, and only linked dashboards + * that the user has permission to update will be synced. + * + * To **link** or **unlink** a UDD set the `lookml_link_id` property with [update_dashboard()](#!/Dashboard/update_dashboard) + * + */ + @Override + public void syncLookmlDashboard(SyncLookmlDashboardReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/dashboards/{lookml_dashboard_id}/sync", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SyncLookmlDashboardResponse.Builder responseBuilder = SyncLookmlDashboardResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about a dashboard + * + * Returns the full details of the identified dashboard object + * + * Get a **summary list** of all active dashboards with [all_dashboards()](#!/Dashboard/all_dashboards) + * + * You can **Search** for dashboards with [search_dashboards()](#!/Dashboard/search_dashboards) + * + */ + @Override + public void dashboard(DashboardReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/dashboards/{dashboard_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DashboardResponse.Builder responseBuilder = DashboardResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update a dashboard + * + * You can use this function to change the string and integer properties of + * a dashboard. Nested objects such as filters, dashboard elements, or dashboard layout components + * cannot be modified by this function - use the update functions for the respective + * nested object types (like [update_dashboard_filter()](#!/3.1/Dashboard/update_dashboard_filter) to change a filter) + * to modify nested objects referenced by a dashboard. + * + * If you receive a 422 error response when updating a dashboard, be sure to look at the + * response body for information about exactly which fields are missing or contain invalid data. + * + */ + @Override + public void updateDashboard(UpdateDashboardReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/dashboards/{dashboard_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateDashboardResponse.Builder responseBuilder = UpdateDashboardResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Delete the dashboard with the specified id + * + * Permanently **deletes** a dashboard. (The dashboard cannot be recovered after this operation.) + * + * "Soft" delete or hide a dashboard by setting its `deleted` status to `True` with [update_dashboard()](#!/Dashboard/update_dashboard). + * + * Note: When a dashboard is deleted in the UI, it is soft deleted. Use this API call to permanently remove it, if desired. + * + */ + @Override + public void deleteDashboard(DeleteDashboardReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/dashboards/{dashboard_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteDashboardResponse.Builder responseBuilder = DeleteDashboardResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get Aggregate Table LookML for Each Query on a Dahboard + * + * Returns a JSON object that contains the dashboard id and Aggregate Table lookml + * + * + */ + @Override + public void dashboardAggregateTableLookml(DashboardAggregateTableLookmlReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/dashboards/aggregate_table_lookml/{dashboard_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DashboardAggregateTableLookmlResponse.Builder responseBuilder = DashboardAggregateTableLookmlResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get lookml of a UDD + * + * Returns a JSON object that contains the dashboard id and the full lookml + * + * + */ + @Override + public void dashboardLookml(DashboardLookmlReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/dashboards/lookml/{dashboard_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DashboardLookmlResponse.Builder responseBuilder = DashboardLookmlResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Move an existing dashboard + * + * Moves a dashboard to a specified folder, and returns the moved dashboard. + * + * `dashboard_id` and `folder_id` are required. + * `dashboard_id` and `folder_id` must already exist, and `folder_id` must be different from the current `folder_id` of the dashboard. + * + */ + @Override + public void moveDashboard(MoveDashboardReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/dashboards/{dashboard_id}/move", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + MoveDashboardResponse.Builder responseBuilder = MoveDashboardResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Creates a new dashboard object based on LookML Dashboard YAML, and returns the details of the newly created dashboard. + * + * This is equivalent to creating a LookML Dashboard and converting to a User-defined dashboard. + * + * LookML must contain valid LookML YAML code. It's recommended to use the LookML format returned + * from [dashboard_lookml()](#!/Dashboard/dashboard_lookml) as the input LookML (newlines replaced with + * ). + * + * Note that the created dashboard is not linked to any LookML Dashboard, + * i.e. [sync_lookml_dashboard()](#!/Dashboard/sync_lookml_dashboard) will not update dashboards created by this method. + * + * + */ + @Override + public void createDashboardFromLookml(CreateDashboardFromLookmlReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/dashboards/from_lookml", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateDashboardFromLookmlResponse.Builder responseBuilder = CreateDashboardFromLookmlResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Copy an existing dashboard + * + * Creates a copy of an existing dashboard, in a specified folder, and returns the copied dashboard. + * + * `dashboard_id` is required, `dashboard_id` and `folder_id` must already exist if specified. + * `folder_id` will default to the existing folder. + * + * If a dashboard with the same title already exists in the target folder, the copy will have '(copy)' + * or '(copy <# of copies>)' appended. + * + */ + @Override + public void copyDashboard(CopyDashboardReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/dashboards/{dashboard_id}/copy", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CopyDashboardResponse.Builder responseBuilder = CopyDashboardResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Search Dashboard Elements + * + * Returns an **array of DashboardElement objects** that match the specified search criteria. + * + * If multiple search params are given and `filter_or` is FALSE or not specified, + * search params are combined in a logical AND operation. + * Only rows that match *all* search param criteria will be returned. + * + * If `filter_or` is TRUE, multiple search params are combined in a logical OR operation. + * Results will include rows that match **any** of the search criteria. + * + * String search params use case-insensitive matching. + * String search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions. + * example="dan%" will match "danger" and "Danzig" but not "David" + * example="D_m%" will match "Damage" and "dump" + * + * Integer search params can accept a single value or a comma separated list of values. The multiple + * values will be combined under a logical OR operation - results will match at least one of + * the given values. + * + * Most search params can accept "IS NULL" and "NOT NULL" as special expressions to match + * or exclude (respectively) rows where the column is null. + * + * Boolean search params accept only "true" and "false" as values. + * + * + */ + @Override + public void searchDashboardElements(SearchDashboardElementsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/dashboard_elements/search", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SearchDashboardElementsResponse.Builder responseBuilder = SearchDashboardElementsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about the dashboard element with a specific id. + */ + @Override + public void dashboardElement(DashboardElementReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/dashboard_elements/{dashboard_element_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DashboardElementResponse.Builder responseBuilder = DashboardElementResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update the dashboard element with a specific id. + */ + @Override + public void updateDashboardElement(UpdateDashboardElementReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/dashboard_elements/{dashboard_element_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateDashboardElementResponse.Builder responseBuilder = UpdateDashboardElementResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Delete a dashboard element with a specific id. + */ + @Override + public void deleteDashboardElement(DeleteDashboardElementReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/dashboard_elements/{dashboard_element_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteDashboardElementResponse.Builder responseBuilder = DeleteDashboardElementResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about all the dashboard elements on a dashboard with a specific id. + */ + @Override + public void dashboardDashboardElements(DashboardDashboardElementsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/dashboards/{dashboard_id}/dashboard_elements", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DashboardDashboardElementsResponse.Builder responseBuilder = DashboardDashboardElementsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create a dashboard element on the dashboard with a specific id. + */ + @Override + public void createDashboardElement(CreateDashboardElementReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/dashboard_elements", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateDashboardElementResponse.Builder responseBuilder = CreateDashboardElementResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about the dashboard filters with a specific id. + */ + @Override + public void dashboardFilter(DashboardFilterReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/dashboard_filters/{dashboard_filter_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DashboardFilterResponse.Builder responseBuilder = DashboardFilterResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update the dashboard filter with a specific id. + */ + @Override + public void updateDashboardFilter(UpdateDashboardFilterReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/dashboard_filters/{dashboard_filter_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateDashboardFilterResponse.Builder responseBuilder = UpdateDashboardFilterResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Delete a dashboard filter with a specific id. + */ + @Override + public void deleteDashboardFilter(DeleteDashboardFilterReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/dashboard_filters/{dashboard_filter_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteDashboardFilterResponse.Builder responseBuilder = DeleteDashboardFilterResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about all the dashboard filters on a dashboard with a specific id. + */ + @Override + public void dashboardDashboardFilters(DashboardDashboardFiltersReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/dashboards/{dashboard_id}/dashboard_filters", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DashboardDashboardFiltersResponse.Builder responseBuilder = DashboardDashboardFiltersResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create a dashboard filter on the dashboard with a specific id. + */ + @Override + public void createDashboardFilter(CreateDashboardFilterReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/dashboard_filters", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateDashboardFilterResponse.Builder responseBuilder = CreateDashboardFilterResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about the dashboard elements with a specific id. + */ + @Override + public void dashboardLayoutComponent(DashboardLayoutComponentReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/dashboard_layout_components/{dashboard_layout_component_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DashboardLayoutComponentResponse.Builder responseBuilder = DashboardLayoutComponentResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update the dashboard element with a specific id. + */ + @Override + public void updateDashboardLayoutComponent(UpdateDashboardLayoutComponentReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/dashboard_layout_components/{dashboard_layout_component_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateDashboardLayoutComponentResponse.Builder responseBuilder = UpdateDashboardLayoutComponentResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about all the dashboard layout components for a dashboard layout with a specific id. + */ + @Override + public void dashboardLayoutDashboardLayoutComponents(DashboardLayoutDashboardLayoutComponentsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/dashboard_layouts/{dashboard_layout_id}/dashboard_layout_components", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DashboardLayoutDashboardLayoutComponentsResponse.Builder responseBuilder = DashboardLayoutDashboardLayoutComponentsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about the dashboard layouts with a specific id. + */ + @Override + public void dashboardLayout(DashboardLayoutReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/dashboard_layouts/{dashboard_layout_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DashboardLayoutResponse.Builder responseBuilder = DashboardLayoutResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update the dashboard layout with a specific id. + */ + @Override + public void updateDashboardLayout(UpdateDashboardLayoutReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/dashboard_layouts/{dashboard_layout_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateDashboardLayoutResponse.Builder responseBuilder = UpdateDashboardLayoutResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Delete a dashboard layout with a specific id. + */ + @Override + public void deleteDashboardLayout(DeleteDashboardLayoutReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/dashboard_layouts/{dashboard_layout_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteDashboardLayoutResponse.Builder responseBuilder = DeleteDashboardLayoutResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about all the dashboard elements on a dashboard with a specific id. + */ + @Override + public void dashboardDashboardLayouts(DashboardDashboardLayoutsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/dashboards/{dashboard_id}/dashboard_layouts", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DashboardDashboardLayoutsResponse.Builder responseBuilder = DashboardDashboardLayoutsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create a dashboard layout on the dashboard with a specific id. + */ + @Override + public void createDashboardLayout(CreateDashboardLayoutReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/dashboard_layouts", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateDashboardLayoutResponse.Builder responseBuilder = CreateDashboardLayoutResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + //#endregion Dashboard: Manage Dashboards + + //#region DataAction: Run Data Actions + + /** + * Perform a data action. The data action object can be obtained from query results, and used to perform an arbitrary action. + */ + @Override + public void performDataAction(PerformDataActionReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/data_actions", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + PerformDataActionResponse.Builder responseBuilder = PerformDataActionResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * For some data actions, the remote server may supply a form requesting further user input. This endpoint takes a data action, asks the remote server to generate a form for it, and returns that form to you for presentation to the user. + */ + @Override + public void fetchRemoteDataActionForm(FetchRemoteDataActionFormReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/data_actions/form", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + FetchRemoteDataActionFormResponse.Builder responseBuilder = FetchRemoteDataActionFormResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + //#endregion DataAction: Run Data Actions + + //#region Datagroup: Manage Datagroups + + /** + * ### Get information about all datagroups. + * + */ + @Override + public void allDatagroups(AllDatagroupsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/datagroups", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllDatagroupsResponse.Builder responseBuilder = AllDatagroupsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about a datagroup. + * + */ + @Override + public void datagroup(DatagroupReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/datagroups/{datagroup_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DatagroupResponse.Builder responseBuilder = DatagroupResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update a datagroup using the specified params. + * + */ + @Override + public void updateDatagroup(UpdateDatagroupReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/datagroups/{datagroup_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateDatagroupResponse.Builder responseBuilder = UpdateDatagroupResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + //#endregion Datagroup: Manage Datagroups + + //#region DerivedTable: View Derived Table graphs + + /** + * ### Discover information about derived tables + * + */ + @Override + public void graphDerivedTablesForModel(GraphDerivedTablesForModelReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/derived_table/graph/model/{model}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + GraphDerivedTablesForModelResponse.Builder responseBuilder = GraphDerivedTablesForModelResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get the subgraph representing this derived table and its dependencies. + * + */ + @Override + public void graphDerivedTablesForView(GraphDerivedTablesForViewReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/derived_table/graph/view/{view}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + GraphDerivedTablesForViewResponse.Builder responseBuilder = GraphDerivedTablesForViewResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * Enqueue materialization for a PDT with the given model name and view name + */ + @Override + public void startPdtBuild(StartPdtBuildReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/derived_table/{model_name}/{view_name}/start", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + StartPdtBuildResponse.Builder responseBuilder = StartPdtBuildResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * Check status of PDT materialization + */ + @Override + public void checkPdtBuild(CheckPdtBuildReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/derived_table/{materialization_id}/status", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CheckPdtBuildResponse.Builder responseBuilder = CheckPdtBuildResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * Stop a PDT materialization + */ + @Override + public void stopPdtBuild(StopPdtBuildReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/derived_table/{materialization_id}/stop", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + StopPdtBuildResponse.Builder responseBuilder = StopPdtBuildResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + //#endregion DerivedTable: View Derived Table graphs + + //#region Folder: Manage Folders + + /** + * Search for folders by creator id, parent id, name, etc + */ + @Override + public void searchFolders(SearchFoldersReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/folders/search", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SearchFoldersResponse.Builder responseBuilder = SearchFoldersResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about the folder with a specific id. + */ + @Override + public void folder(FolderReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/folders/{folder_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + FolderResponse.Builder responseBuilder = FolderResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update the folder with a specific id. + */ + @Override + public void updateFolder(UpdateFolderReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/folders/{folder_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateFolderResponse.Builder responseBuilder = UpdateFolderResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Delete the folder with a specific id including any children folders. + * **DANGER** this will delete all looks and dashboards in the folder. + * + */ + @Override + public void deleteFolder(DeleteFolderReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/folders/{folder_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteFolderResponse.Builder responseBuilder = DeleteFolderResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about all folders. + * + * In API 3.x, this will not return empty personal folders, unless they belong to the calling user, + * or if they contain soft-deleted content. + * + * In API 4.0+, all personal folders will be returned. + * + * + */ + @Override + public void allFolders(AllFoldersReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/folders", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllFoldersResponse.Builder responseBuilder = AllFoldersResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create a folder with specified information. + * + * Caller must have permission to edit the parent folder and to create folders, otherwise the request + * returns 404 Not Found. + * + */ + @Override + public void createFolder(CreateFolderReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/folders", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateFolderResponse.Builder responseBuilder = CreateFolderResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get the children of a folder. + */ + @Override + public void folderChildren(FolderChildrenReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/folders/{folder_id}/children", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + FolderChildrenResponse.Builder responseBuilder = FolderChildrenResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Search the children of a folder + */ + @Override + public void folderChildrenSearch(FolderChildrenSearchReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/folders/{folder_id}/children/search", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + FolderChildrenSearchResponse.Builder responseBuilder = FolderChildrenSearchResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get the parent of a folder + */ + @Override + public void folderParent(FolderParentReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/folders/{folder_id}/parent", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + FolderParentResponse.Builder responseBuilder = FolderParentResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get the ancestors of a folder + */ + @Override + public void folderAncestors(FolderAncestorsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/folders/{folder_id}/ancestors", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + FolderAncestorsResponse.Builder responseBuilder = FolderAncestorsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get all looks in a folder. + * In API 3.x, this will return all looks in a folder, including looks in the trash. + * In API 4.0+, all looks in a folder will be returned, excluding looks in the trash. + * + */ + @Override + public void folderLooks(FolderLooksReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/folders/{folder_id}/looks", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + FolderLooksResponse.Builder responseBuilder = FolderLooksResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get the dashboards in a folder + */ + @Override + public void folderDashboards(FolderDashboardsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/folders/{folder_id}/dashboards", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + FolderDashboardsResponse.Builder responseBuilder = FolderDashboardsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + //#endregion Folder: Manage Folders + + //#region Group: Manage Groups + + /** + * ### Get information about all groups. + * + */ + @Override + public void allGroups(AllGroupsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/groups", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllGroupsResponse.Builder responseBuilder = AllGroupsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Creates a new group (admin only). + * + */ + @Override + public void createGroup(CreateGroupReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/groups", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateGroupResponse.Builder responseBuilder = CreateGroupResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Search groups + * + * Returns all group records that match the given search criteria. + * + * If multiple search params are given and `filter_or` is FALSE or not specified, + * search params are combined in a logical AND operation. + * Only rows that match *all* search param criteria will be returned. + * + * If `filter_or` is TRUE, multiple search params are combined in a logical OR operation. + * Results will include rows that match **any** of the search criteria. + * + * String search params use case-insensitive matching. + * String search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions. + * example="dan%" will match "danger" and "Danzig" but not "David" + * example="D_m%" will match "Damage" and "dump" + * + * Integer search params can accept a single value or a comma separated list of values. The multiple + * values will be combined under a logical OR operation - results will match at least one of + * the given values. + * + * Most search params can accept "IS NULL" and "NOT NULL" as special expressions to match + * or exclude (respectively) rows where the column is null. + * + * Boolean search params accept only "true" and "false" as values. + * + * + */ + @Override + public void searchGroups(SearchGroupsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/groups/search", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SearchGroupsResponse.Builder responseBuilder = SearchGroupsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Search groups include roles + * + * Returns all group records that match the given search criteria, and attaches any associated roles. + * + * If multiple search params are given and `filter_or` is FALSE or not specified, + * search params are combined in a logical AND operation. + * Only rows that match *all* search param criteria will be returned. + * + * If `filter_or` is TRUE, multiple search params are combined in a logical OR operation. + * Results will include rows that match **any** of the search criteria. + * + * String search params use case-insensitive matching. + * String search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions. + * example="dan%" will match "danger" and "Danzig" but not "David" + * example="D_m%" will match "Damage" and "dump" + * + * Integer search params can accept a single value or a comma separated list of values. The multiple + * values will be combined under a logical OR operation - results will match at least one of + * the given values. + * + * Most search params can accept "IS NULL" and "NOT NULL" as special expressions to match + * or exclude (respectively) rows where the column is null. + * + * Boolean search params accept only "true" and "false" as values. + * + * + */ + @Override + public void searchGroupsWithRoles(SearchGroupsWithRolesReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/groups/search/with_roles", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SearchGroupsWithRolesResponse.Builder responseBuilder = SearchGroupsWithRolesResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Search groups include hierarchy + * + * Returns all group records that match the given search criteria, and attaches + * associated role_ids and parent group_ids. + * + * If multiple search params are given and `filter_or` is FALSE or not specified, + * search params are combined in a logical AND operation. + * Only rows that match *all* search param criteria will be returned. + * + * If `filter_or` is TRUE, multiple search params are combined in a logical OR operation. + * Results will include rows that match **any** of the search criteria. + * + * String search params use case-insensitive matching. + * String search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions. + * example="dan%" will match "danger" and "Danzig" but not "David" + * example="D_m%" will match "Damage" and "dump" + * + * Integer search params can accept a single value or a comma separated list of values. The multiple + * values will be combined under a logical OR operation - results will match at least one of + * the given values. + * + * Most search params can accept "IS NULL" and "NOT NULL" as special expressions to match + * or exclude (respectively) rows where the column is null. + * + * Boolean search params accept only "true" and "false" as values. + * + * + */ + @Override + public void searchGroupsWithHierarchy(SearchGroupsWithHierarchyReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/groups/search/with_hierarchy", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SearchGroupsWithHierarchyResponse.Builder responseBuilder = SearchGroupsWithHierarchyResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about a group. + * + */ + @Override + public void group(GroupReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/groups/{group_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + GroupResponse.Builder responseBuilder = GroupResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Updates the a group (admin only). + */ + @Override + public void updateGroup(UpdateGroupReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/groups/{group_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateGroupResponse.Builder responseBuilder = UpdateGroupResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Deletes a group (admin only). + * + */ + @Override + public void deleteGroup(DeleteGroupReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/groups/{group_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteGroupResponse.Builder responseBuilder = DeleteGroupResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about all the groups in a group + * + */ + @Override + public void allGroupGroups(AllGroupGroupsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/groups/{group_id}/groups", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllGroupGroupsResponse.Builder responseBuilder = AllGroupGroupsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Adds a new group to a group. + * + */ + @Override + public void addGroupGroup(AddGroupGroupReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/groups/{group_id}/groups", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AddGroupGroupResponse.Builder responseBuilder = AddGroupGroupResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about all the users directly included in a group. + * + */ + @Override + public void allGroupUsers(AllGroupUsersReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/groups/{group_id}/users", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllGroupUsersResponse.Builder responseBuilder = AllGroupUsersResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Adds a new user to a group. + * + */ + @Override + public void addGroupUser(AddGroupUserReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/groups/{group_id}/users", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AddGroupUserResponse.Builder responseBuilder = AddGroupUserResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Removes a user from a group. + * + */ + @Override + public void deleteGroupUser(DeleteGroupUserReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/groups/{group_id}/users/{user_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteGroupUserResponse.Builder responseBuilder = DeleteGroupUserResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Removes a group from a group. + * + */ + @Override + public void deleteGroupFromGroup(DeleteGroupFromGroupReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/groups/{group_id}/groups/{deleting_group_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteGroupFromGroupResponse.Builder responseBuilder = DeleteGroupFromGroupResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Set the value of a user attribute for a group. + * + * For information about how user attribute values are calculated, see [Set User Attribute Group Values](#!/UserAttribute/set_user_attribute_group_values). + * + */ + @Override + public void updateUserAttributeGroupValue(UpdateUserAttributeGroupValueReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/groups/{group_id}/attribute_values/{user_attribute_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateUserAttributeGroupValueResponse.Builder responseBuilder = UpdateUserAttributeGroupValueResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Remove a user attribute value from a group. + * + */ + @Override + public void deleteUserAttributeGroupValue(DeleteUserAttributeGroupValueReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/groups/{group_id}/attribute_values/{user_attribute_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteUserAttributeGroupValueResponse.Builder responseBuilder = DeleteUserAttributeGroupValueResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + //#endregion Group: Manage Groups + + //#region Homepage: Manage Homepage + + /** + * ### Get information about the primary homepage's sections. + * + */ + @Override + public void allPrimaryHomepageSections(AllPrimaryHomepageSectionsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/primary_homepage_sections", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllPrimaryHomepageSectionsResponse.Builder responseBuilder = AllPrimaryHomepageSectionsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + //#endregion Homepage: Manage Homepage + + //#region Integration: Manage Integrations + + /** + * ### Get information about all Integration Hubs. + * + */ + @Override + public void allIntegrationHubs(AllIntegrationHubsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/integration_hubs", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllIntegrationHubsResponse.Builder responseBuilder = AllIntegrationHubsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create a new Integration Hub. + * + * This API is rate limited to prevent it from being used for SSRF attacks + * + */ + @Override + public void createIntegrationHub(CreateIntegrationHubReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/integration_hubs", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateIntegrationHubResponse.Builder responseBuilder = CreateIntegrationHubResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about a Integration Hub. + * + */ + @Override + public void integrationHub(IntegrationHubReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/integration_hubs/{integration_hub_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + IntegrationHubResponse.Builder responseBuilder = IntegrationHubResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update a Integration Hub definition. + * + * This API is rate limited to prevent it from being used for SSRF attacks + * + */ + @Override + public void updateIntegrationHub(UpdateIntegrationHubReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/integration_hubs/{integration_hub_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateIntegrationHubResponse.Builder responseBuilder = UpdateIntegrationHubResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Delete a Integration Hub. + * + */ + @Override + public void deleteIntegrationHub(DeleteIntegrationHubReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/integration_hubs/{integration_hub_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteIntegrationHubResponse.Builder responseBuilder = DeleteIntegrationHubResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * Accepts the legal agreement for a given integration hub. This only works for integration hubs that have legal_agreement_required set to true and legal_agreement_signed set to false. + */ + @Override + public void acceptIntegrationHubLegalAgreement(AcceptIntegrationHubLegalAgreementReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/integration_hubs/{integration_hub_id}/accept_legal_agreement", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AcceptIntegrationHubLegalAgreementResponse.Builder responseBuilder = AcceptIntegrationHubLegalAgreementResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about all Integrations. + * + */ + @Override + public void allIntegrations(AllIntegrationsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/integrations", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllIntegrationsResponse.Builder responseBuilder = AllIntegrationsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about a Integration. + * + */ + @Override + public void integration(IntegrationReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/integrations/{integration_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + IntegrationResponse.Builder responseBuilder = IntegrationResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update parameters on a Integration. + * + */ + @Override + public void updateIntegration(UpdateIntegrationReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/integrations/{integration_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateIntegrationResponse.Builder responseBuilder = UpdateIntegrationResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * Returns the Integration form for presentation to the user. + */ + @Override + public void fetchIntegrationForm(FetchIntegrationFormReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/integrations/{integration_id}/form", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + FetchIntegrationFormResponse.Builder responseBuilder = FetchIntegrationFormResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * Tests the integration to make sure all the settings are working. + */ + @Override + public void testIntegration(TestIntegrationReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/integrations/{integration_id}/test", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + TestIntegrationResponse.Builder responseBuilder = TestIntegrationResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + //#endregion Integration: Manage Integrations + + //#region Look: Run and Manage Looks + + /** + * ### Get information about all active Looks + * + * Returns an array of **abbreviated Look objects** describing all the looks that the caller has access to. Soft-deleted Looks are **not** included. + * + * Get the **full details** of a specific look by id with [look(id)](#!/Look/look) + * + * Find **soft-deleted looks** with [search_looks()](#!/Look/search_looks) + * + */ + @Override + public void allLooks(AllLooksReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/looks", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllLooksResponse.Builder responseBuilder = AllLooksResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create a Look + * + * To create a look to display query data, first create the query with [create_query()](#!/Query/create_query) + * then assign the query's id to the `query_id` property in the call to `create_look()`. + * + * To place the look into a particular space, assign the space's id to the `space_id` property + * in the call to `create_look()`. + * + */ + @Override + public void createLook(CreateLookReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/looks", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateLookResponse.Builder responseBuilder = CreateLookResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Search Looks + * + * Returns an **array of Look objects** that match the specified search criteria. + * + * If multiple search params are given and `filter_or` is FALSE or not specified, + * search params are combined in a logical AND operation. + * Only rows that match *all* search param criteria will be returned. + * + * If `filter_or` is TRUE, multiple search params are combined in a logical OR operation. + * Results will include rows that match **any** of the search criteria. + * + * String search params use case-insensitive matching. + * String search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions. + * example="dan%" will match "danger" and "Danzig" but not "David" + * example="D_m%" will match "Damage" and "dump" + * + * Integer search params can accept a single value or a comma separated list of values. The multiple + * values will be combined under a logical OR operation - results will match at least one of + * the given values. + * + * Most search params can accept "IS NULL" and "NOT NULL" as special expressions to match + * or exclude (respectively) rows where the column is null. + * + * Boolean search params accept only "true" and "false" as values. + * + * + * Get a **single look** by id with [look(id)](#!/Look/look) + * + */ + @Override + public void searchLooks(SearchLooksReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/looks/search", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SearchLooksResponse.Builder responseBuilder = SearchLooksResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get a Look. + * + * Returns detailed information about a Look and its associated Query. + * + * + */ + @Override + public void look(LookReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/looks/{look_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + LookResponse.Builder responseBuilder = LookResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Modify a Look + * + * Use this function to modify parts of a look. Property values given in a call to `update_look` are + * applied to the existing look, so there's no need to include properties whose values are not changing. + * It's best to specify only the properties you want to change and leave everything else out + * of your `update_look` call. **Look properties marked 'read-only' will be ignored.** + * + * When a user deletes a look in the Looker UI, the look data remains in the database but is + * marked with a deleted flag ("soft-deleted"). Soft-deleted looks can be undeleted (by an admin) + * if the delete was in error. + * + * To soft-delete a look via the API, use [update_look()](#!/Look/update_look) to change the look's `deleted` property to `true`. + * You can undelete a look by calling `update_look` to change the look's `deleted` property to `false`. + * + * Soft-deleted looks are excluded from the results of [all_looks()](#!/Look/all_looks) and [search_looks()](#!/Look/search_looks), so they + * essentially disappear from view even though they still reside in the db. + * In API 3.1 and later, you can pass `deleted: true` as a parameter to [search_looks()](#!/3.1/Look/search_looks) to list soft-deleted looks. + * + * NOTE: [delete_look()](#!/Look/delete_look) performs a "hard delete" - the look data is removed from the Looker + * database and destroyed. There is no "undo" for `delete_look()`. + * + */ + @Override + public void updateLook(UpdateLookReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/looks/{look_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateLookResponse.Builder responseBuilder = UpdateLookResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Permanently Delete a Look + * + * This operation **permanently** removes a look from the Looker database. + * + * NOTE: There is no "undo" for this kind of delete. + * + * For information about soft-delete (which can be undone) see [update_look()](#!/Look/update_look). + * + */ + @Override + public void deleteLook(DeleteLookReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/looks/{look_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteLookResponse.Builder responseBuilder = DeleteLookResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Run a Look + * + * Runs a given look's query and returns the results in the requested format. + * + * Supported formats: + * + * | result_format | Description + * | :-----------: | :--- | + * | json | Plain json + * | json_detail | Row data plus metadata describing the fields, pivots, table calcs, and other aspects of the query + * | csv | Comma separated values with a header + * | txt | Tab separated values with a header + * | html | Simple html + * | md | Simple markdown + * | xlsx | MS Excel spreadsheet + * | sql | Returns the generated SQL rather than running the query + * | png | A PNG image of the visualization of the query + * | jpg | A JPG image of the visualization of the query + * + * + * + */ + @Override + public void runLook(RunLookReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/looks/{look_id}/run/{result_format}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + RunLookResponse.Builder responseBuilder = RunLookResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Copy an existing look + * + * Creates a copy of an existing look, in a specified folder, and returns the copied look. + * + * `look_id` and `folder_id` are required. + * + * `look_id` and `folder_id` must already exist, and `folder_id` must be different from the current `folder_id` of the dashboard. + * + */ + @Override + public void copyLook(CopyLookReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/looks/{look_id}/copy", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CopyLookResponse.Builder responseBuilder = CopyLookResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Move an existing look + * + * Moves a look to a specified folder, and returns the moved look. + * + * `look_id` and `folder_id` are required. + * `look_id` and `folder_id` must already exist, and `folder_id` must be different from the current `folder_id` of the dashboard. + * + */ + @Override + public void moveLook(MoveLookReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/looks/{look_id}/move", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + MoveLookResponse.Builder responseBuilder = MoveLookResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + //#endregion Look: Run and Manage Looks + + //#region LookmlModel: Manage LookML Models + + /** + * ### Get information about all lookml models. + * + */ + @Override + public void allLookmlModels(AllLookmlModelsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/lookml_models", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllLookmlModelsResponse.Builder responseBuilder = AllLookmlModelsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create a lookml model using the specified configuration. + * + */ + @Override + public void createLookmlModel(CreateLookmlModelReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/lookml_models", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateLookmlModelResponse.Builder responseBuilder = CreateLookmlModelResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about a lookml model. + * + */ + @Override + public void lookmlModel(LookmlModelReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/lookml_models/{lookml_model_name}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + LookmlModelResponse.Builder responseBuilder = LookmlModelResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update a lookml model using the specified configuration. + * + */ + @Override + public void updateLookmlModel(UpdateLookmlModelReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/lookml_models/{lookml_model_name}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateLookmlModelResponse.Builder responseBuilder = UpdateLookmlModelResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Delete a lookml model. + * + */ + @Override + public void deleteLookmlModel(DeleteLookmlModelReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/lookml_models/{lookml_model_name}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteLookmlModelResponse.Builder responseBuilder = DeleteLookmlModelResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about a lookml model explore. + * + */ + @Override + public void lookmlModelExplore(LookmlModelExploreReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/lookml_models/{lookml_model_name}/explores/{explore_name}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + LookmlModelExploreResponse.Builder responseBuilder = LookmlModelExploreResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + //#endregion LookmlModel: Manage LookML Models + + //#region Metadata: Connection Metadata Features + + /** + * ### Field name suggestions for a model and view + * + * `filters` is a string hash of values, with the key as the field name and the string value as the filter expression: + * + * ```ruby + * {'users.age': '>=60'} + * ``` + * + * or + * + * ```ruby + * {'users.age': '<30'} + * ``` + * + * or + * + * ```ruby + * {'users.age': '=50'} + * ``` + * + */ + @Override + public void modelFieldnameSuggestions(ModelFieldnameSuggestionsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/models/{model_name}/views/{view_name}/fields/{field_name}/suggestions", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ModelFieldnameSuggestionsResponse.Builder responseBuilder = ModelFieldnameSuggestionsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get a single model + * + * + */ + @Override + public void getModel(GetModelReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/models/{model_name}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + GetModelResponse.Builder responseBuilder = GetModelResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### List databases available to this connection + * + * Certain dialects can support multiple databases per single connection. + * If this connection supports multiple databases, the database names will be returned in an array. + * + * Connections using dialects that do not support multiple databases will return an empty array. + * + * **Note**: [Connection Features](#!/Metadata/connection_features) can be used to determine if a connection supports + * multiple databases. + * + */ + @Override + public void connectionDatabases(ConnectionDatabasesReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/connections/{connection_name}/databases", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ConnectionDatabasesResponse.Builder responseBuilder = ConnectionDatabasesResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Retrieve metadata features for this connection + * + * Returns a list of feature names with `true` (available) or `false` (not available) + * + * + */ + @Override + public void connectionFeatures(ConnectionFeaturesReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/connections/{connection_name}/features", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ConnectionFeaturesResponse.Builder responseBuilder = ConnectionFeaturesResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get the list of schemas and tables for a connection + * + * + */ + @Override + public void connectionSchemas(ConnectionSchemasReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/connections/{connection_name}/schemas", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ConnectionSchemasResponse.Builder responseBuilder = ConnectionSchemasResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get the list of tables for a schema + * + * For dialects that support multiple databases, optionally identify which to use. If not provided, the default + * database for the connection will be used. + * + * For dialects that do **not** support multiple databases, **do not use** the database parameter + * + */ + @Override + public void connectionTables(ConnectionTablesReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/connections/{connection_name}/tables", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ConnectionTablesResponse.Builder responseBuilder = ConnectionTablesResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get the columns (and therefore also the tables) in a specific schema + * + * + */ + @Override + public void connectionColumns(ConnectionColumnsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/connections/{connection_name}/columns", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ConnectionColumnsResponse.Builder responseBuilder = ConnectionColumnsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Search a connection for columns matching the specified name + * + * **Note**: `column_name` must be a valid column name. It is not a search pattern. + * + */ + @Override + public void connectionSearchColumns(ConnectionSearchColumnsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/connections/{connection_name}/search_columns", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ConnectionSearchColumnsResponse.Builder responseBuilder = ConnectionSearchColumnsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Connection cost estimating + * + * Assign a `sql` statement to the body of the request. e.g., for Ruby, `{sql: 'select * from users'}` + * + * **Note**: If the connection's dialect has no support for cost estimates, an error will be returned + * + */ + @Override + public void connectionCostEstimate(ConnectionCostEstimateReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/connections/{connection_name}/cost_estimate", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ConnectionCostEstimateResponse.Builder responseBuilder = ConnectionCostEstimateResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + //#endregion Metadata: Connection Metadata Features + + //#region Project: Manage Projects + + /** + * ### Generate Lockfile for All LookML Dependencies + * + * Git must have been configured, must be in dev mode and deploy permission required + * + * Install_all is a two step process + * 1. For each remote_dependency in a project the dependency manager will resolve any ambiguous ref. + * 2. The project will then write out a lockfile including each remote_dependency with its resolved ref. + * + * + */ + @Override + public void lockAll(LockAllReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/projects/{project_id}/manifest/lock_all", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + LockAllResponse.Builder responseBuilder = LockAllResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get All Git Branches + * + * Returns a list of git branches in the project repository + * + */ + @Override + public void allGitBranches(AllGitBranchesReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/projects/{project_id}/git_branches", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllGitBranchesResponse.Builder responseBuilder = AllGitBranchesResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get the Current Git Branch + * + * Returns the git branch currently checked out in the given project repository + * + */ + @Override + public void gitBranch(GitBranchReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/projects/{project_id}/git_branch", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + GitBranchResponse.Builder responseBuilder = GitBranchResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Checkout and/or reset --hard an existing Git Branch + * + * Only allowed in development mode + * - Call `update_session` to select the 'dev' workspace. + * + * Checkout an existing branch if name field is different from the name of the currently checked out branch. + * + * Optionally specify a branch name, tag name or commit SHA to which the branch should be reset. + * **DANGER** hard reset will be force pushed to the remote. Unsaved changes and commits may be permanently lost. + * + * + */ + @Override + public void updateGitBranch(UpdateGitBranchReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.put("/projects/{project_id}/git_branch", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateGitBranchResponse.Builder responseBuilder = UpdateGitBranchResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create and Checkout a Git Branch + * + * Creates and checks out a new branch in the given project repository + * Only allowed in development mode + * - Call `update_session` to select the 'dev' workspace. + * + * Optionally specify a branch name, tag name or commit SHA as the start point in the ref field. + * If no ref is specified, HEAD of the current branch will be used as the start point for the new branch. + * + * + */ + @Override + public void createGitBranch(CreateGitBranchReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/projects/{project_id}/git_branch", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateGitBranchResponse.Builder responseBuilder = CreateGitBranchResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get the specified Git Branch + * + * Returns the git branch specified in branch_name path param if it exists in the given project repository + * + */ + @Override + public void findGitBranch(FindGitBranchReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/projects/{project_id}/git_branch/{branch_name}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + FindGitBranchResponse.Builder responseBuilder = FindGitBranchResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Delete the specified Git Branch + * + * Delete git branch specified in branch_name path param from local and remote of specified project repository + * + */ + @Override + public void deleteGitBranch(DeleteGitBranchReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/projects/{project_id}/git_branch/{branch_name}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteGitBranchResponse.Builder responseBuilder = DeleteGitBranchResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Deploy a Remote Branch or Ref to Production + * + * Git must have been configured and deploy permission required. + * + * Deploy is a one/two step process + * 1. If this is the first deploy of this project, create the production project with git repository. + * 2. Pull the branch or ref into the production project. + * + * Can only specify either a branch or a ref. + * + * + */ + @Override + public void deployRefToProduction(DeployRefToProductionReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/projects/{project_id}/deploy_ref_to_production", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeployRefToProductionResponse.Builder responseBuilder = DeployRefToProductionResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Deploy LookML from this Development Mode Project to Production + * + * Git must have been configured, must be in dev mode and deploy permission required + * + * Deploy is a two / three step process: + * + * 1. Push commits in current branch of dev mode project to the production branch (origin/master). + * Note a. This step is skipped in read-only projects. + * Note b. If this step is unsuccessful for any reason (e.g. rejected non-fastforward because production branch has + * commits not in current branch), subsequent steps will be skipped. + * 2. If this is the first deploy of this project, create the production project with git repository. + * 3. Pull the production branch into the production project. + * + * + */ + @Override + public void deployToProduction(DeployToProductionReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/projects/{project_id}/deploy_to_production", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeployToProductionResponse.Builder responseBuilder = DeployToProductionResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Reset a project to the revision of the project that is in production. + * + * **DANGER** this will delete any changes that have not been pushed to a remote repository. + * + */ + @Override + public void resetProjectToProduction(ResetProjectToProductionReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/projects/{project_id}/reset_to_production", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ResetProjectToProductionResponse.Builder responseBuilder = ResetProjectToProductionResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Reset a project development branch to the revision of the project that is on the remote. + * + * **DANGER** this will delete any changes that have not been pushed to a remote repository. + * + */ + @Override + public void resetProjectToRemote(ResetProjectToRemoteReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/projects/{project_id}/reset_to_remote", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ResetProjectToRemoteResponse.Builder responseBuilder = ResetProjectToRemoteResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get All Projects + * + * Returns all projects visible to the current user + * + */ + @Override + public void allProjects(AllProjectsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/projects", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllProjectsResponse.Builder responseBuilder = AllProjectsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create A Project + * + * dev mode required. + * - Call `update_session` to select the 'dev' workspace. + * + * `name` is required. + * `git_remote_url` is not allowed. To configure Git for the newly created project, follow the instructions in `update_project`. + * + * + */ + @Override + public void createProject(CreateProjectReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/projects", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateProjectResponse.Builder responseBuilder = CreateProjectResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get A Project + * + * Returns the project with the given project id + * + */ + @Override + public void project(ProjectReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/projects/{project_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ProjectResponse.Builder responseBuilder = ProjectResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update Project Configuration + * + * Apply changes to a project's configuration. + * + * + * #### Configuring Git for a Project + * + * To set up a Looker project with a remote git repository, follow these steps: + * + * 1. Call `update_session` to select the 'dev' workspace. + * 1. Call `create_git_deploy_key` to create a new deploy key for the project + * 1. Copy the deploy key text into the remote git repository's ssh key configuration + * 1. Call `update_project` to set project's `git_remote_url` ()and `git_service_name`, if necessary). + * + * When you modify a project's `git_remote_url`, Looker connects to the remote repository to fetch + * metadata. The remote git repository MUST be configured with the Looker-generated deploy + * key for this project prior to setting the project's `git_remote_url`. + * + * To set up a Looker project with a git repository residing on the Looker server (a 'bare' git repo): + * + * 1. Call `update_session` to select the 'dev' workspace. + * 1. Call `update_project` setting `git_remote_url` to null and `git_service_name` to "bare". + * + * + */ + @Override + public void updateProject(UpdateProjectReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/projects/{project_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateProjectResponse.Builder responseBuilder = UpdateProjectResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get A Projects Manifest object + * + * Returns the project with the given project id + * + */ + @Override + public void manifest(ManifestReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/projects/{project_id}/manifest", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ManifestResponse.Builder responseBuilder = ManifestResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Git Deploy Key + * + * Returns the ssh public key previously created for a project's git repository. + * + */ + @Override + public void gitDeployKey(GitDeployKeyReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/projects/{project_id}/git/deploy_key", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + GitDeployKeyResponse.Builder responseBuilder = GitDeployKeyResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create Git Deploy Key + * + * Create a public/private key pair for authenticating ssh git requests from Looker to a remote git repository + * for a particular Looker project. + * + * Returns the public key of the generated ssh key pair. + * + * Copy this public key to your remote git repository's ssh keys configuration so that the remote git service can + * validate and accept git requests from the Looker server. + * + */ + @Override + public void createGitDeployKey(CreateGitDeployKeyReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/projects/{project_id}/git/deploy_key", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateGitDeployKeyResponse.Builder responseBuilder = CreateGitDeployKeyResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get Cached Project Validation Results + * + * Returns the cached results of a previous project validation calculation, if any. + * Returns http status 204 No Content if no validation results exist. + * + * Validating the content of all the files in a project can be computationally intensive + * for large projects. Use this API to simply fetch the results of the most recent + * project validation rather than revalidating the entire project from scratch. + * + * A value of `"stale": true` in the response indicates that the project has changed since + * the cached validation results were computed. The cached validation results may no longer + * reflect the current state of the project. + * + */ + @Override + public void projectValidationResults(ProjectValidationResultsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/projects/{project_id}/validate", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ProjectValidationResultsResponse.Builder responseBuilder = ProjectValidationResultsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Validate Project + * + * Performs lint validation of all lookml files in the project. + * Returns a list of errors found, if any. + * + * Validating the content of all the files in a project can be computationally intensive + * for large projects. For best performance, call `validate_project(project_id)` only + * when you really want to recompute project validation. To quickly display the results of + * the most recent project validation (without recomputing), use `project_validation_results(project_id)` + * + */ + @Override + public void validateProject(ValidateProjectReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/projects/{project_id}/validate", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ValidateProjectResponse.Builder responseBuilder = ValidateProjectResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get Project Workspace + * + * Returns information about the state of the project files in the currently selected workspace + * + */ + @Override + public void projectWorkspace(ProjectWorkspaceReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/projects/{project_id}/current_workspace", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ProjectWorkspaceResponse.Builder responseBuilder = ProjectWorkspaceResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get All Project Files + * + * Returns a list of the files in the project + * + */ + @Override + public void allProjectFiles(AllProjectFilesReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/projects/{project_id}/files", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllProjectFilesResponse.Builder responseBuilder = AllProjectFilesResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get Project File Info + * + * Returns information about a file in the project + * + */ + @Override + public void projectFile(ProjectFileReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/projects/{project_id}/files/file", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ProjectFileResponse.Builder responseBuilder = ProjectFileResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get All Git Connection Tests + * + * dev mode required. + * - Call `update_session` to select the 'dev' workspace. + * + * Returns a list of tests which can be run against a project's (or the dependency project for the provided remote_url) git connection. Call [Run Git Connection Test](#!/Project/run_git_connection_test) to execute each test in sequence. + * + * Tests are ordered by increasing specificity. Tests should be run in the order returned because later tests require functionality tested by tests earlier in the test list. + * + * For example, a late-stage test for write access is meaningless if connecting to the git server (an early test) is failing. + * + */ + @Override + public void allGitConnectionTests(AllGitConnectionTestsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/projects/{project_id}/git_connection_tests", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllGitConnectionTestsResponse.Builder responseBuilder = AllGitConnectionTestsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Run a git connection test + * + * Run the named test on the git service used by this project (or the dependency project for the provided remote_url) and return the result. This + * is intended to help debug git connections when things do not work properly, to give + * more helpful information about why a git url is not working with Looker. + * + * Tests should be run in the order they are returned by [Get All Git Connection Tests](#!/Project/all_git_connection_tests). + * + */ + @Override + public void runGitConnectionTest(RunGitConnectionTestReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/projects/{project_id}/git_connection_tests/{test_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + RunGitConnectionTestResponse.Builder responseBuilder = RunGitConnectionTestResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get All LookML Tests + * + * Returns a list of tests which can be run to validate a project's LookML code and/or the underlying data, + * optionally filtered by the file id. + * Call [Run LookML Test](#!/Project/run_lookml_test) to execute tests. + * + */ + @Override + public void allLookmlTests(AllLookmlTestsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/projects/{project_id}/lookml_tests", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllLookmlTestsResponse.Builder responseBuilder = AllLookmlTestsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Run LookML Tests + * + * Runs all tests in the project, optionally filtered by file, test, and/or model. + * + */ + @Override + public void runLookmlTest(RunLookmlTestReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/projects/{project_id}/lookml_tests/run", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + RunLookmlTestResponse.Builder responseBuilder = RunLookmlTestResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Creates a tag for the most recent commit, or a specific ref is a SHA is provided + * + * This is an internal-only, undocumented route. + * + */ + @Override + public void tagRef(TagRefReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/projects/{project_id}/tag", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + TagRefResponse.Builder responseBuilder = TagRefResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Configure Repository Credential for a remote dependency + * + * Admin required. + * + * `root_project_id` is required. + * `credential_id` is required. + * + * + */ + @Override + public void updateRepositoryCredential(UpdateRepositoryCredentialReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.put("/projects/{root_project_id}/credential/{credential_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateRepositoryCredentialResponse.Builder responseBuilder = UpdateRepositoryCredentialResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Repository Credential for a remote dependency + * + * Admin required. + * + * `root_project_id` is required. + * `credential_id` is required. + * + */ + @Override + public void deleteRepositoryCredential(DeleteRepositoryCredentialReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/projects/{root_project_id}/credential/{credential_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteRepositoryCredentialResponse.Builder responseBuilder = DeleteRepositoryCredentialResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get all Repository Credentials for a project + * + * `root_project_id` is required. + * + */ + @Override + public void getAllRepositoryCredentials(GetAllRepositoryCredentialsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/projects/{root_project_id}/credentials", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + GetAllRepositoryCredentialsResponse.Builder responseBuilder = GetAllRepositoryCredentialsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + //#endregion Project: Manage Projects + + //#region Query: Run and Manage Queries + + /** + * ### Create an async query task + * + * Creates a query task (job) to run a previously created query asynchronously. Returns a Query Task ID. + * + * Use [query_task(query_task_id)](#!/Query/query_task) to check the execution status of the query task. + * After the query task status reaches "Complete", use [query_task_results(query_task_id)](#!/Query/query_task_results) to fetch the results of the query. + * + */ + @Override + public void createQueryTask(CreateQueryTaskReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/query_tasks", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateQueryTaskResponse.Builder responseBuilder = CreateQueryTaskResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Fetch results of multiple async queries + * + * Returns the results of multiple async queries in one request. + * + * For Query Tasks that are not completed, the response will include the execution status of the Query Task but will not include query results. + * Query Tasks whose results have expired will have a status of 'expired'. + * If the user making the API request does not have sufficient privileges to view a Query Task result, the result will have a status of 'missing' + * + */ + @Override + public void queryTaskMultiResults(QueryTaskMultiResultsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/query_tasks/multi_results", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + QueryTaskMultiResultsResponse.Builder responseBuilder = QueryTaskMultiResultsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get Query Task details + * + * Use this function to check the status of an async query task. After the status + * reaches "Complete", you can call [query_task_results(query_task_id)](#!/Query/query_task_results) to + * retrieve the results of the query. + * + * Use [create_query_task()](#!/Query/create_query_task) to create an async query task. + * + */ + @Override + public void queryTask(QueryTaskReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/query_tasks/{query_task_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + QueryTaskResponse.Builder responseBuilder = QueryTaskResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get Async Query Results + * + * Returns the results of an async query task if the query has completed. + * + * If the query task is still running or waiting to run, this function returns 204 No Content. + * + * If the query task ID is invalid or the cached results of the query task have expired, this function returns 404 Not Found. + * + * Use [query_task(query_task_id)](#!/Query/query_task) to check the execution status of the query task + * Call query_task_results only after the query task status reaches "Complete". + * + * You can also use [query_task_multi_results()](#!/Query/query_task_multi_results) retrieve the + * results of multiple async query tasks at the same time. + * + * #### SQL Error Handling: + * If the query fails due to a SQL db error, how this is communicated depends on the result_format you requested in `create_query_task()`. + * + * For `json_detail` result_format: `query_task_results()` will respond with HTTP status '200 OK' and db SQL error info + * will be in the `errors` property of the response object. The 'data' property will be empty. + * + * For all other result formats: `query_task_results()` will respond with HTTP status `400 Bad Request` and some db SQL error info + * will be in the message of the 400 error response, but not as detailed as expressed in `json_detail.errors`. + * These data formats can only carry row data, and error info is not row data. + * + */ + @Override + public void queryTaskResults(QueryTaskResultsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/query_tasks/{query_task_id}/results", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + QueryTaskResultsResponse.Builder responseBuilder = QueryTaskResultsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get a previously created query by id. + * + * A Looker query object includes the various parameters that define a database query that has been run or + * could be run in the future. These parameters include: model, view, fields, filters, pivots, etc. + * Query *results* are not part of the query object. + * + * Query objects are unique and immutable. Query objects are created automatically in Looker as users explore data. + * Looker does not delete them; they become part of the query history. When asked to create a query for + * any given set of parameters, Looker will first try to find an existing query object with matching + * parameters and will only create a new object when an appropriate object can not be found. + * + * This 'get' method is used to get the details about a query for a given id. See the other methods here + * to 'create' and 'run' queries. + * + * Note that some fields like 'filter_config' and 'vis_config' etc are specific to how the Looker UI + * builds queries and visualizations and are not generally useful for API use. They are not required when + * creating new queries and can usually just be ignored. + * + * + */ + @Override + public void query(QueryReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/queries/{query_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + QueryResponse.Builder responseBuilder = QueryResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get the query for a given query slug. + * + * This returns the query for the 'slug' in a query share URL. + * + * The 'slug' is a randomly chosen short string that is used as an alternative to the query's id value + * for use in URLs etc. This method exists as a convenience to help you use the API to 'find' queries that + * have been created using the Looker UI. + * + * You can use the Looker explore page to build a query and then choose the 'Share' option to + * show the share url for the query. Share urls generally look something like 'https://looker.yourcompany/x/vwGSbfc'. + * The trailing 'vwGSbfc' is the share slug. You can pass that string to this api method to get details about the query. + * Those details include the 'id' that you can use to run the query. Or, you can copy the query body + * (perhaps with your own modification) and use that as the basis to make/run new queries. + * + * This will also work with slugs from Looker explore urls like + * 'https://looker.yourcompany/explore/ecommerce/orders?qid=aogBgL6o3cKK1jN3RoZl5s'. In this case + * 'aogBgL6o3cKK1jN3RoZl5s' is the slug. + * + */ + @Override + public void queryForSlug(QueryForSlugReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/queries/slug/{slug}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + QueryForSlugResponse.Builder responseBuilder = QueryForSlugResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create a query. + * + * This allows you to create a new query that you can later run. Looker queries are immutable once created + * and are not deleted. If you create a query that is exactly like an existing query then the existing query + * will be returned and no new query will be created. Whether a new query is created or not, you can use + * the 'id' in the returned query with the 'run' method. + * + * The query parameters are passed as json in the body of the request. + * + * + */ + @Override + public void createQuery(CreateQueryReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/queries", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateQueryResponse.Builder responseBuilder = CreateQueryResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Run a saved query. + * + * This runs a previously saved query. You can use this on a query that was generated in the Looker UI + * or one that you have explicitly created using the API. You can also use a query 'id' from a saved 'Look'. + * + * The 'result_format' parameter specifies the desired structure and format of the response. + * + * Supported formats: + * + * | result_format | Description + * | :-----------: | :--- | + * | json | Plain json + * | json_detail | Row data plus metadata describing the fields, pivots, table calcs, and other aspects of the query + * | csv | Comma separated values with a header + * | txt | Tab separated values with a header + * | html | Simple html + * | md | Simple markdown + * | xlsx | MS Excel spreadsheet + * | sql | Returns the generated SQL rather than running the query + * | png | A PNG image of the visualization of the query + * | jpg | A JPG image of the visualization of the query + * + * + * + */ + @Override + public void runQuery(RunQueryReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/queries/{query_id}/run/{result_format}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + RunQueryResponse.Builder responseBuilder = RunQueryResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Run the query that is specified inline in the posted body. + * + * This allows running a query as defined in json in the posted body. This combines + * the two actions of posting & running a query into one step. + * + * Here is an example body in json: + * ``` + * { + * "model":"thelook", + * "view":"inventory_items", + * "fields":["category.name","inventory_items.days_in_inventory_tier","products.count"], + * "filters":{"category.name":"socks"}, + * "sorts":["products.count desc 0"], + * "limit":"500", + * "query_timezone":"America/Los_Angeles" + * } + * ``` + * + * When using the Ruby SDK this would be passed as a Ruby hash like: + * ``` + * { + * :model=>"thelook", + * :view=>"inventory_items", + * :fields=> + * ["category.name", + * "inventory_items.days_in_inventory_tier", + * "products.count"], + * :filters=>{:"category.name"=>"socks"}, + * :sorts=>["products.count desc 0"], + * :limit=>"500", + * :query_timezone=>"America/Los_Angeles", + * } + * ``` + * + * This will return the result of running the query in the format specified by the 'result_format' parameter. + * + * Supported formats: + * + * | result_format | Description + * | :-----------: | :--- | + * | json | Plain json + * | json_detail | Row data plus metadata describing the fields, pivots, table calcs, and other aspects of the query + * | csv | Comma separated values with a header + * | txt | Tab separated values with a header + * | html | Simple html + * | md | Simple markdown + * | xlsx | MS Excel spreadsheet + * | sql | Returns the generated SQL rather than running the query + * | png | A PNG image of the visualization of the query + * | jpg | A JPG image of the visualization of the query + * + * + * + */ + @Override + public void runInlineQuery(RunInlineQueryReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/queries/run/{result_format}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + RunInlineQueryResponse.Builder responseBuilder = RunInlineQueryResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Run an URL encoded query. + * + * This requires the caller to encode the specifiers for the query into the URL query part using + * Looker-specific syntax as explained below. + * + * Generally, you would want to use one of the methods that takes the parameters as json in the POST body + * for creating and/or running queries. This method exists for cases where one really needs to encode the + * parameters into the URL of a single 'GET' request. This matches the way that the Looker UI formats + * 'explore' URLs etc. + * + * The parameters here are very similar to the json body formatting except that the filter syntax is + * tricky. Unfortunately, this format makes this method not currently callable via the 'Try it out!' button + * in this documentation page. But, this is callable when creating URLs manually or when using the Looker SDK. + * + * Here is an example inline query URL: + * + * ``` + * https://looker.mycompany.com:19999/api/3.0/queries/models/thelook/views/inventory_items/run/json?fields=category.name,inventory_items.days_in_inventory_tier,products.count&f[category.name]=socks&sorts=products.count+desc+0&limit=500&query_timezone=America/Los_Angeles + * ``` + * + * When invoking this endpoint with the Ruby SDK, pass the query parameter parts as a hash. The hash to match the above would look like: + * + * ```ruby + * query_params = + * { + * fields: "category.name,inventory_items.days_in_inventory_tier,products.count", + * :"f[category.name]" => "socks", + * sorts: "products.count desc 0", + * limit: "500", + * query_timezone: "America/Los_Angeles" + * } + * response = ruby_sdk.run_url_encoded_query('thelook','inventory_items','json', query_params) + * + * ``` + * + * Again, it is generally easier to use the variant of this method that passes the full query in the POST body. + * This method is available for cases where other alternatives won't fit the need. + * + * Supported formats: + * + * | result_format | Description + * | :-----------: | :--- | + * | json | Plain json + * | json_detail | Row data plus metadata describing the fields, pivots, table calcs, and other aspects of the query + * | csv | Comma separated values with a header + * | txt | Tab separated values with a header + * | html | Simple html + * | md | Simple markdown + * | xlsx | MS Excel spreadsheet + * | sql | Returns the generated SQL rather than running the query + * | png | A PNG image of the visualization of the query + * | jpg | A JPG image of the visualization of the query + * + * + * + */ + @Override + public void runUrlEncodedQuery(RunUrlEncodedQueryReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/queries/models/{model_name}/views/{view_name}/run/{result_format}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + RunUrlEncodedQueryResponse.Builder responseBuilder = RunUrlEncodedQueryResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get Merge Query + * + * Returns a merge query object given its id. + * + */ + @Override + public void mergeQuery(MergeQueryReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/merge_queries/{merge_query_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + MergeQueryResponse.Builder responseBuilder = MergeQueryResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create Merge Query + * + * Creates a new merge query object. + * + * A merge query takes the results of one or more queries and combines (merges) the results + * according to field mapping definitions. The result is similar to a SQL left outer join. + * + * A merge query can merge results of queries from different SQL databases. + * + * The order that queries are defined in the source_queries array property is significant. The + * first query in the array defines the primary key into which the results of subsequent + * queries will be merged. + * + * Like model/view query objects, merge queries are immutable and have structural identity - if + * you make a request to create a new merge query that is identical to an existing merge query, + * the existing merge query will be returned instead of creating a duplicate. Conversely, any + * change to the contents of a merge query will produce a new object with a new id. + * + */ + @Override + public void createMergeQuery(CreateMergeQueryReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/merge_queries", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateMergeQueryResponse.Builder responseBuilder = CreateMergeQueryResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * Get information about all running queries. + * + */ + @Override + public void allRunningQueries(AllRunningQueriesReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/running_queries", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllRunningQueriesResponse.Builder responseBuilder = AllRunningQueriesResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * Kill a query with a specific query_task_id. + * + */ + @Override + public void killQuery(KillQueryReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/running_queries/{query_task_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + KillQueryResponse.Builder responseBuilder = KillQueryResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * Get a SQL Runner query. + */ + @Override + public void sqlQuery(SqlQueryReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/sql_queries/{slug}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SqlQueryResponse.Builder responseBuilder = SqlQueryResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create a SQL Runner Query + * + * Either the `connection_name` or `model_name` parameter MUST be provided. + * + */ + @Override + public void createSqlQuery(CreateSqlQueryReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/sql_queries", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateSqlQueryResponse.Builder responseBuilder = CreateSqlQueryResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * Execute a SQL Runner query in a given result_format. + */ + @Override + public void runSqlQuery(RunSqlQueryReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/sql_queries/{slug}/run/{result_format}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + RunSqlQueryResponse.Builder responseBuilder = RunSqlQueryResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + //#endregion Query: Run and Manage Queries + + //#region RenderTask: Manage Render Tasks + + /** + * ### Create a new task to render a look to an image. + * + * Returns a render task object. + * To check the status of a render task, pass the render_task.id to [Get Render Task](#!/RenderTask/get_render_task). + * Once the render task is complete, you can download the resulting document or image using [Get Render Task Results](#!/RenderTask/get_render_task_results). + * + * + */ + @Override + public void createLookRenderTask(CreateLookRenderTaskReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/render_tasks/looks/{look_id}/{result_format}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateLookRenderTaskResponse.Builder responseBuilder = CreateLookRenderTaskResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create a new task to render an existing query to an image. + * + * Returns a render task object. + * To check the status of a render task, pass the render_task.id to [Get Render Task](#!/RenderTask/get_render_task). + * Once the render task is complete, you can download the resulting document or image using [Get Render Task Results](#!/RenderTask/get_render_task_results). + * + * + */ + @Override + public void createQueryRenderTask(CreateQueryRenderTaskReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/render_tasks/queries/{query_id}/{result_format}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateQueryRenderTaskResponse.Builder responseBuilder = CreateQueryRenderTaskResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create a new task to render a dashboard to a document or image. + * + * Returns a render task object. + * To check the status of a render task, pass the render_task.id to [Get Render Task](#!/RenderTask/get_render_task). + * Once the render task is complete, you can download the resulting document or image using [Get Render Task Results](#!/RenderTask/get_render_task_results). + * + * + */ + @Override + public void createDashboardRenderTask(CreateDashboardRenderTaskReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/render_tasks/dashboards/{dashboard_id}/{result_format}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateDashboardRenderTaskResponse.Builder responseBuilder = CreateDashboardRenderTaskResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about a render task. + * + * Returns a render task object. + * To check the status of a render task, pass the render_task.id to [Get Render Task](#!/RenderTask/get_render_task). + * Once the render task is complete, you can download the resulting document or image using [Get Render Task Results](#!/RenderTask/get_render_task_results). + * + * + */ + @Override + public void renderTask(RenderTaskReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/render_tasks/{render_task_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + RenderTaskResponse.Builder responseBuilder = RenderTaskResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get the document or image produced by a completed render task. + * + * Note that the PDF or image result will be a binary blob in the HTTP response, as indicated by the + * Content-Type in the response headers. This may require specialized (or at least different) handling than text + * responses such as JSON. You may need to tell your HTTP client that the response is binary so that it does not + * attempt to parse the binary data as text. + * + * If the render task exists but has not finished rendering the results, the response HTTP status will be + * **202 Accepted**, the response body will be empty, and the response will have a Retry-After header indicating + * that the caller should repeat the request at a later time. + * + * Returns 404 if the render task cannot be found, if the cached result has expired, or if the caller + * does not have permission to view the results. + * + * For detailed information about the status of the render task, use [Render Task](#!/RenderTask/render_task). + * Polling loops waiting for completion of a render task would be better served by polling **render_task(id)** until + * the task status reaches completion (or error) instead of polling **render_task_results(id)** alone. + * + */ + @Override + public void renderTaskResults(RenderTaskResultsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/render_tasks/{render_task_id}/results", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + RenderTaskResultsResponse.Builder responseBuilder = RenderTaskResultsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create a new task to render a dashboard element to an image. + * + * Returns a render task object. + * To check the status of a render task, pass the render_task.id to [Get Render Task](#!/RenderTask/get_render_task). + * Once the render task is complete, you can download the resulting document or image using [Get Render Task Results](#!/RenderTask/get_render_task_results). + * + * + */ + @Override + public void createDashboardElementRenderTask(CreateDashboardElementRenderTaskReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/render_tasks/dashboard_elements/{dashboard_element_id}/{result_format}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateDashboardElementRenderTaskResponse.Builder responseBuilder = CreateDashboardElementRenderTaskResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + //#endregion RenderTask: Manage Render Tasks + + //#region Role: Manage Roles + + /** + * ### Search model sets + * Returns all model set records that match the given search criteria. + * If multiple search params are given and `filter_or` is FALSE or not specified, + * search params are combined in a logical AND operation. + * Only rows that match *all* search param criteria will be returned. + * + * If `filter_or` is TRUE, multiple search params are combined in a logical OR operation. + * Results will include rows that match **any** of the search criteria. + * + * String search params use case-insensitive matching. + * String search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions. + * example="dan%" will match "danger" and "Danzig" but not "David" + * example="D_m%" will match "Damage" and "dump" + * + * Integer search params can accept a single value or a comma separated list of values. The multiple + * values will be combined under a logical OR operation - results will match at least one of + * the given values. + * + * Most search params can accept "IS NULL" and "NOT NULL" as special expressions to match + * or exclude (respectively) rows where the column is null. + * + * Boolean search params accept only "true" and "false" as values. + * + * + */ + @Override + public void searchModelSets(SearchModelSetsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/model_sets/search", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SearchModelSetsResponse.Builder responseBuilder = SearchModelSetsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about the model set with a specific id. + * + */ + @Override + public void modelSet(ModelSetReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/model_sets/{model_set_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ModelSetResponse.Builder responseBuilder = ModelSetResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update information about the model set with a specific id. + * + */ + @Override + public void updateModelSet(UpdateModelSetReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/model_sets/{model_set_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateModelSetResponse.Builder responseBuilder = UpdateModelSetResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Delete the model set with a specific id. + * + */ + @Override + public void deleteModelSet(DeleteModelSetReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/model_sets/{model_set_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteModelSetResponse.Builder responseBuilder = DeleteModelSetResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about all model sets. + * + */ + @Override + public void allModelSets(AllModelSetsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/model_sets", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllModelSetsResponse.Builder responseBuilder = AllModelSetsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create a model set with the specified information. Model sets are used by Roles. + * + */ + @Override + public void createModelSet(CreateModelSetReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/model_sets", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateModelSetResponse.Builder responseBuilder = CreateModelSetResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get all supported permissions. + * + */ + @Override + public void allPermissions(AllPermissionsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/permissions", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllPermissionsResponse.Builder responseBuilder = AllPermissionsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Search permission sets + * Returns all permission set records that match the given search criteria. + * If multiple search params are given and `filter_or` is FALSE or not specified, + * search params are combined in a logical AND operation. + * Only rows that match *all* search param criteria will be returned. + * + * If `filter_or` is TRUE, multiple search params are combined in a logical OR operation. + * Results will include rows that match **any** of the search criteria. + * + * String search params use case-insensitive matching. + * String search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions. + * example="dan%" will match "danger" and "Danzig" but not "David" + * example="D_m%" will match "Damage" and "dump" + * + * Integer search params can accept a single value or a comma separated list of values. The multiple + * values will be combined under a logical OR operation - results will match at least one of + * the given values. + * + * Most search params can accept "IS NULL" and "NOT NULL" as special expressions to match + * or exclude (respectively) rows where the column is null. + * + * Boolean search params accept only "true" and "false" as values. + * + * + */ + @Override + public void searchPermissionSets(SearchPermissionSetsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/permission_sets/search", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SearchPermissionSetsResponse.Builder responseBuilder = SearchPermissionSetsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about the permission set with a specific id. + * + */ + @Override + public void permissionSet(PermissionSetReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/permission_sets/{permission_set_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + PermissionSetResponse.Builder responseBuilder = PermissionSetResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update information about the permission set with a specific id. + * + */ + @Override + public void updatePermissionSet(UpdatePermissionSetReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/permission_sets/{permission_set_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdatePermissionSetResponse.Builder responseBuilder = UpdatePermissionSetResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Delete the permission set with a specific id. + * + */ + @Override + public void deletePermissionSet(DeletePermissionSetReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/permission_sets/{permission_set_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeletePermissionSetResponse.Builder responseBuilder = DeletePermissionSetResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about all permission sets. + * + */ + @Override + public void allPermissionSets(AllPermissionSetsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/permission_sets", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllPermissionSetsResponse.Builder responseBuilder = AllPermissionSetsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create a permission set with the specified information. Permission sets are used by Roles. + * + */ + @Override + public void createPermissionSet(CreatePermissionSetReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/permission_sets", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreatePermissionSetResponse.Builder responseBuilder = CreatePermissionSetResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about all roles. + * + */ + @Override + public void allRoles(AllRolesReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/roles", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllRolesResponse.Builder responseBuilder = AllRolesResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create a role with the specified information. + * + */ + @Override + public void createRole(CreateRoleReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/roles", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateRoleResponse.Builder responseBuilder = CreateRoleResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Search roles + * + * Returns all role records that match the given search criteria. + * + * If multiple search params are given and `filter_or` is FALSE or not specified, + * search params are combined in a logical AND operation. + * Only rows that match *all* search param criteria will be returned. + * + * If `filter_or` is TRUE, multiple search params are combined in a logical OR operation. + * Results will include rows that match **any** of the search criteria. + * + * String search params use case-insensitive matching. + * String search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions. + * example="dan%" will match "danger" and "Danzig" but not "David" + * example="D_m%" will match "Damage" and "dump" + * + * Integer search params can accept a single value or a comma separated list of values. The multiple + * values will be combined under a logical OR operation - results will match at least one of + * the given values. + * + * Most search params can accept "IS NULL" and "NOT NULL" as special expressions to match + * or exclude (respectively) rows where the column is null. + * + * Boolean search params accept only "true" and "false" as values. + * + * + */ + @Override + public void searchRoles(SearchRolesReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/roles/search", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SearchRolesResponse.Builder responseBuilder = SearchRolesResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Search roles include user count + * + * Returns all role records that match the given search criteria, and attaches + * associated user counts. + * + * If multiple search params are given and `filter_or` is FALSE or not specified, + * search params are combined in a logical AND operation. + * Only rows that match *all* search param criteria will be returned. + * + * If `filter_or` is TRUE, multiple search params are combined in a logical OR operation. + * Results will include rows that match **any** of the search criteria. + * + * String search params use case-insensitive matching. + * String search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions. + * example="dan%" will match "danger" and "Danzig" but not "David" + * example="D_m%" will match "Damage" and "dump" + * + * Integer search params can accept a single value or a comma separated list of values. The multiple + * values will be combined under a logical OR operation - results will match at least one of + * the given values. + * + * Most search params can accept "IS NULL" and "NOT NULL" as special expressions to match + * or exclude (respectively) rows where the column is null. + * + * Boolean search params accept only "true" and "false" as values. + * + * + */ + @Override + public void searchRolesWithUserCount(SearchRolesWithUserCountReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/roles/search/with_user_count", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SearchRolesWithUserCountResponse.Builder responseBuilder = SearchRolesWithUserCountResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about the role with a specific id. + * + */ + @Override + public void role(RoleReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/roles/{role_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + RoleResponse.Builder responseBuilder = RoleResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update information about the role with a specific id. + * + */ + @Override + public void updateRole(UpdateRoleReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/roles/{role_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateRoleResponse.Builder responseBuilder = UpdateRoleResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Delete the role with a specific id. + * + */ + @Override + public void deleteRole(DeleteRoleReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/roles/{role_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteRoleResponse.Builder responseBuilder = DeleteRoleResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about all the groups with the role that has a specific id. + * + */ + @Override + public void roleGroups(RoleGroupsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/roles/{role_id}/groups", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + RoleGroupsResponse.Builder responseBuilder = RoleGroupsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Set all groups for a role, removing all existing group associations from that role. + * + */ + @Override + public void setRoleGroups(SetRoleGroupsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.put("/roles/{role_id}/groups", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SetRoleGroupsResponse.Builder responseBuilder = SetRoleGroupsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about all the users with the role that has a specific id. + * + */ + @Override + public void roleUsers(RoleUsersReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/roles/{role_id}/users", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + RoleUsersResponse.Builder responseBuilder = RoleUsersResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Set all the users of the role with a specific id. + * + */ + @Override + public void setRoleUsers(SetRoleUsersReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.put("/roles/{role_id}/users", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SetRoleUsersResponse.Builder responseBuilder = SetRoleUsersResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + //#endregion Role: Manage Roles + + //#region ScheduledPlan: Manage Scheduled Plans + + /** + * ### Get Scheduled Plans for a Space + * + * Returns scheduled plans owned by the caller for a given space id. + * + */ + @Override + public void scheduledPlansForSpace(ScheduledPlansForSpaceReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/scheduled_plans/space/{space_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ScheduledPlansForSpaceResponse.Builder responseBuilder = ScheduledPlansForSpaceResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get Information About a Scheduled Plan + * + * Admins can fetch information about other users' Scheduled Plans. + * + */ + @Override + public void scheduledPlan(ScheduledPlanReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/scheduled_plans/{scheduled_plan_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ScheduledPlanResponse.Builder responseBuilder = ScheduledPlanResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update a Scheduled Plan + * + * Admins can update other users' Scheduled Plans. + * + * Note: Any scheduled plan destinations specified in an update will **replace** all scheduled plan destinations + * currently defined for the scheduled plan. + * + * For Example: If a scheduled plan has destinations A, B, and C, and you call update on this scheduled plan + * specifying only B in the destinations, then destinations A and C will be deleted by the update. + * + * Updating a scheduled plan to assign null or an empty array to the scheduled_plan_destinations property is an error, as a scheduled plan must always have at least one destination. + * + * If you omit the scheduled_plan_destinations property from the object passed to update, then the destinations + * defined on the original scheduled plan will remain unchanged. + * + * #### Email Permissions: + * + * For details about permissions required to schedule delivery to email and the safeguards + * Looker offers to protect against sending to unauthorized email destinations, see [Email Domain Whitelist for Scheduled Looks](https://docs.looker.com/r/api/embed-permissions). + * + * + * #### Scheduled Plan Destination Formats + * + * Scheduled plan destinations must specify the data format to produce and send to the destination. + * + * Formats: + * + * | format | Description + * | :-----------: | :--- | + * | json | A JSON object containing a `data` property which contains an array of JSON objects, one per row. No metadata. + * | json_detail | Row data plus metadata describing the fields, pivots, table calcs, and other aspects of the query + * | inline_json | Same as the JSON format, except that the `data` property is a string containing JSON-escaped row data. Additional properties describe the data operation. This format is primarily used to send data to web hooks so that the web hook doesn't have to re-encode the JSON row data in order to pass it on to its ultimate destination. + * | csv | Comma separated values with a header + * | txt | Tab separated values with a header + * | html | Simple html + * | xlsx | MS Excel spreadsheet + * | wysiwyg_pdf | Dashboard rendered in a tiled layout to produce a PDF document + * | assembled_pdf | Dashboard rendered in a single column layout to produce a PDF document + * | wysiwyg_png | Dashboard rendered in a tiled layout to produce a PNG image + * || + * + * Valid formats vary by destination type and source object. `wysiwyg_pdf` is only valid for dashboards, for example. + * + * + * + */ + @Override + public void updateScheduledPlan(UpdateScheduledPlanReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/scheduled_plans/{scheduled_plan_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateScheduledPlanResponse.Builder responseBuilder = UpdateScheduledPlanResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Delete a Scheduled Plan + * + * Normal users can only delete their own scheduled plans. + * Admins can delete other users' scheduled plans. + * This delete cannot be undone. + * + */ + @Override + public void deleteScheduledPlan(DeleteScheduledPlanReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/scheduled_plans/{scheduled_plan_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteScheduledPlanResponse.Builder responseBuilder = DeleteScheduledPlanResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### List All Scheduled Plans + * + * Returns all scheduled plans which belong to the caller or given user. + * + * If no user_id is provided, this function returns the scheduled plans owned by the caller. + * + * + * To list all schedules for all users, pass `all_users=true`. + * + * + * The caller must have `see_schedules` permission to see other users' scheduled plans. + * + * + * + */ + @Override + public void allScheduledPlans(AllScheduledPlansReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/scheduled_plans", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllScheduledPlansResponse.Builder responseBuilder = AllScheduledPlansResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create a Scheduled Plan + * + * Create a scheduled plan to render a Look or Dashboard on a recurring schedule. + * + * To create a scheduled plan, you MUST provide values for the following fields: + * `name` + * and + * `look_id`, `dashboard_id`, `lookml_dashboard_id`, or `query_id` + * and + * `cron_tab` or `datagroup` + * and + * at least one scheduled_plan_destination + * + * A scheduled plan MUST have at least one scheduled_plan_destination defined. + * + * When `look_id` is set, `require_no_results`, `require_results`, and `require_change` are all required. + * + * If `create_scheduled_plan` fails with a 422 error, be sure to look at the error messages in the response which will explain exactly what fields are missing or values that are incompatible. + * + * The queries that provide the data for the look or dashboard are run in the context of user account that owns the scheduled plan. + * + * When `run_as_recipient` is `false` or not specified, the queries that provide the data for the + * look or dashboard are run in the context of user account that owns the scheduled plan. + * + * When `run_as_recipient` is `true` and all the email recipients are Looker user accounts, the + * queries are run in the context of each recipient, so different recipients may see different + * data from the same scheduled render of a look or dashboard. For more details, see [Run As Recipient](https://looker.com/docs/r/admin/run-as-recipient). + * + * Admins can create and modify scheduled plans on behalf of other users by specifying a user id. + * Non-admin users may not create or modify scheduled plans by or for other users. + * + * #### Email Permissions: + * + * For details about permissions required to schedule delivery to email and the safeguards + * Looker offers to protect against sending to unauthorized email destinations, see [Email Domain Whitelist for Scheduled Looks](https://docs.looker.com/r/api/embed-permissions). + * + * + * #### Scheduled Plan Destination Formats + * + * Scheduled plan destinations must specify the data format to produce and send to the destination. + * + * Formats: + * + * | format | Description + * | :-----------: | :--- | + * | json | A JSON object containing a `data` property which contains an array of JSON objects, one per row. No metadata. + * | json_detail | Row data plus metadata describing the fields, pivots, table calcs, and other aspects of the query + * | inline_json | Same as the JSON format, except that the `data` property is a string containing JSON-escaped row data. Additional properties describe the data operation. This format is primarily used to send data to web hooks so that the web hook doesn't have to re-encode the JSON row data in order to pass it on to its ultimate destination. + * | csv | Comma separated values with a header + * | txt | Tab separated values with a header + * | html | Simple html + * | xlsx | MS Excel spreadsheet + * | wysiwyg_pdf | Dashboard rendered in a tiled layout to produce a PDF document + * | assembled_pdf | Dashboard rendered in a single column layout to produce a PDF document + * | wysiwyg_png | Dashboard rendered in a tiled layout to produce a PNG image + * || + * + * Valid formats vary by destination type and source object. `wysiwyg_pdf` is only valid for dashboards, for example. + * + * + * + */ + @Override + public void createScheduledPlan(CreateScheduledPlanReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/scheduled_plans", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateScheduledPlanResponse.Builder responseBuilder = CreateScheduledPlanResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Run a Scheduled Plan Immediately + * + * Create a scheduled plan that runs only once, and immediately. + * + * This can be useful for testing a Scheduled Plan before committing to a production schedule. + * + * Admins can create scheduled plans on behalf of other users by specifying a user id. + * + * This API is rate limited to prevent it from being used for relay spam or DoS attacks + * + * #### Email Permissions: + * + * For details about permissions required to schedule delivery to email and the safeguards + * Looker offers to protect against sending to unauthorized email destinations, see [Email Domain Whitelist for Scheduled Looks](https://docs.looker.com/r/api/embed-permissions). + * + * + * #### Scheduled Plan Destination Formats + * + * Scheduled plan destinations must specify the data format to produce and send to the destination. + * + * Formats: + * + * | format | Description + * | :-----------: | :--- | + * | json | A JSON object containing a `data` property which contains an array of JSON objects, one per row. No metadata. + * | json_detail | Row data plus metadata describing the fields, pivots, table calcs, and other aspects of the query + * | inline_json | Same as the JSON format, except that the `data` property is a string containing JSON-escaped row data. Additional properties describe the data operation. This format is primarily used to send data to web hooks so that the web hook doesn't have to re-encode the JSON row data in order to pass it on to its ultimate destination. + * | csv | Comma separated values with a header + * | txt | Tab separated values with a header + * | html | Simple html + * | xlsx | MS Excel spreadsheet + * | wysiwyg_pdf | Dashboard rendered in a tiled layout to produce a PDF document + * | assembled_pdf | Dashboard rendered in a single column layout to produce a PDF document + * | wysiwyg_png | Dashboard rendered in a tiled layout to produce a PNG image + * || + * + * Valid formats vary by destination type and source object. `wysiwyg_pdf` is only valid for dashboards, for example. + * + * + * + */ + @Override + public void scheduledPlanRunOnce(ScheduledPlanRunOnceReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/scheduled_plans/run_once", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ScheduledPlanRunOnceResponse.Builder responseBuilder = ScheduledPlanRunOnceResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get Scheduled Plans for a Look + * + * Returns all scheduled plans for a look which belong to the caller or given user. + * + * If no user_id is provided, this function returns the scheduled plans owned by the caller. + * + * + * To list all schedules for all users, pass `all_users=true`. + * + * + * The caller must have `see_schedules` permission to see other users' scheduled plans. + * + * + * + */ + @Override + public void scheduledPlansForLook(ScheduledPlansForLookReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/scheduled_plans/look/{look_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ScheduledPlansForLookResponse.Builder responseBuilder = ScheduledPlansForLookResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get Scheduled Plans for a Dashboard + * + * Returns all scheduled plans for a dashboard which belong to the caller or given user. + * + * If no user_id is provided, this function returns the scheduled plans owned by the caller. + * + * + * To list all schedules for all users, pass `all_users=true`. + * + * + * The caller must have `see_schedules` permission to see other users' scheduled plans. + * + * + * + */ + @Override + public void scheduledPlansForDashboard(ScheduledPlansForDashboardReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/scheduled_plans/dashboard/{dashboard_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ScheduledPlansForDashboardResponse.Builder responseBuilder = ScheduledPlansForDashboardResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get Scheduled Plans for a LookML Dashboard + * + * Returns all scheduled plans for a LookML Dashboard which belong to the caller or given user. + * + * If no user_id is provided, this function returns the scheduled plans owned by the caller. + * + * + * To list all schedules for all users, pass `all_users=true`. + * + * + * The caller must have `see_schedules` permission to see other users' scheduled plans. + * + * + * + */ + @Override + public void scheduledPlansForLookmlDashboard(ScheduledPlansForLookmlDashboardReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/scheduled_plans/lookml_dashboard/{lookml_dashboard_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ScheduledPlansForLookmlDashboardResponse.Builder responseBuilder = ScheduledPlansForLookmlDashboardResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Run a Scheduled Plan By Id Immediately + * This function creates a run-once schedule plan based on an existing scheduled plan, + * applies modifications (if any) to the new scheduled plan, and runs the new schedule plan immediately. + * This can be useful for testing modifications to an existing scheduled plan before committing to a production schedule. + * + * This function internally performs the following operations: + * + * 1. Copies the properties of the existing scheduled plan into a new scheduled plan + * 2. Copies any properties passed in the JSON body of this request into the new scheduled plan (replacing the original values) + * 3. Creates the new scheduled plan + * 4. Runs the new scheduled plan + * + * The original scheduled plan is not modified by this operation. + * Admins can create, modify, and run scheduled plans on behalf of other users by specifying a user id. + * Non-admins can only create, modify, and run their own scheduled plans. + * + * #### Email Permissions: + * + * For details about permissions required to schedule delivery to email and the safeguards + * Looker offers to protect against sending to unauthorized email destinations, see [Email Domain Whitelist for Scheduled Looks](https://docs.looker.com/r/api/embed-permissions). + * + * + * #### Scheduled Plan Destination Formats + * + * Scheduled plan destinations must specify the data format to produce and send to the destination. + * + * Formats: + * + * | format | Description + * | :-----------: | :--- | + * | json | A JSON object containing a `data` property which contains an array of JSON objects, one per row. No metadata. + * | json_detail | Row data plus metadata describing the fields, pivots, table calcs, and other aspects of the query + * | inline_json | Same as the JSON format, except that the `data` property is a string containing JSON-escaped row data. Additional properties describe the data operation. This format is primarily used to send data to web hooks so that the web hook doesn't have to re-encode the JSON row data in order to pass it on to its ultimate destination. + * | csv | Comma separated values with a header + * | txt | Tab separated values with a header + * | html | Simple html + * | xlsx | MS Excel spreadsheet + * | wysiwyg_pdf | Dashboard rendered in a tiled layout to produce a PDF document + * | assembled_pdf | Dashboard rendered in a single column layout to produce a PDF document + * | wysiwyg_png | Dashboard rendered in a tiled layout to produce a PNG image + * || + * + * Valid formats vary by destination type and source object. `wysiwyg_pdf` is only valid for dashboards, for example. + * + * + * + * This API is rate limited to prevent it from being used for relay spam or DoS attacks + * + * + */ + @Override + public void scheduledPlanRunOnceById(ScheduledPlanRunOnceByIdReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/scheduled_plans/{scheduled_plan_id}/run_once", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ScheduledPlanRunOnceByIdResponse.Builder responseBuilder = ScheduledPlanRunOnceByIdResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + //#endregion ScheduledPlan: Manage Scheduled Plans + + //#region Session: Session Information + + /** + * ### Get API Session + * + * Returns information about the current API session, such as which workspace is selected for the session. + * + */ + @Override + public void session(SessionReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/session", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SessionResponse.Builder responseBuilder = SessionResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update API Session + * + * #### API Session Workspace + * + * You can use this endpoint to change the active workspace for the current API session. + * + * Only one workspace can be active in a session. The active workspace can be changed + * any number of times in a session. + * + * The default workspace for API sessions is the "production" workspace. + * + * All Looker APIs that use projects or lookml models (such as running queries) will + * use the version of project and model files defined by this workspace for the lifetime of the + * current API session or until the session workspace is changed again. + * + * An API session has the same lifetime as the access_token used to authenticate API requests. Each successful + * API login generates a new access_token and a new API session. + * + * If your Looker API client application needs to work in a dev workspace across multiple + * API sessions, be sure to select the dev workspace after each login. + * + */ + @Override + public void updateSession(UpdateSessionReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/session", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateSessionResponse.Builder responseBuilder = UpdateSessionResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + //#endregion Session: Session Information + + //#region Theme: Manage Themes + + /** + * ### Get an array of all existing themes + * + * Get a **single theme** by id with [Theme](#!/Theme/theme) + * + * This method returns an array of all existing themes. The active time for the theme is not considered. + * + * **Note**: Custom themes needs to be enabled by Looker. Unless custom themes are enabled, only the automatically generated default theme can be used. Please contact your Account Manager or help.looker.com to update your license for this feature. + * + * + */ + @Override + public void allThemes(AllThemesReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/themes", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllThemesResponse.Builder responseBuilder = AllThemesResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create a theme + * + * Creates a new theme object, returning the theme details, including the created id. + * + * If `settings` are not specified, the default theme settings will be copied into the new theme. + * + * The theme `name` can only contain alphanumeric characters or underscores. Theme names should not contain any confidential information, such as customer names. + * + * **Update** an existing theme with [Update Theme](#!/Theme/update_theme) + * + * **Permanently delete** an existing theme with [Delete Theme](#!/Theme/delete_theme) + * + * For more information, see [Creating and Applying Themes](https://looker.com/docs/r/admin/themes). + * + * **Note**: Custom themes needs to be enabled by Looker. Unless custom themes are enabled, only the automatically generated default theme can be used. Please contact your Account Manager or help.looker.com to update your license for this feature. + * + * + */ + @Override + public void createTheme(CreateThemeReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/themes", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateThemeResponse.Builder responseBuilder = CreateThemeResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Search all themes for matching criteria. + * + * Returns an **array of theme objects** that match the specified search criteria. + * + * | Search Parameters | Description + * | :-------------------: | :------ | + * | `begin_at` only | Find themes active at or after `begin_at` + * | `end_at` only | Find themes active at or before `end_at` + * | both set | Find themes with an active inclusive period between `begin_at` and `end_at` + * + * Note: Range matching requires boolean AND logic. + * When using `begin_at` and `end_at` together, do not use `filter_or`=TRUE + * + * If multiple search params are given and `filter_or` is FALSE or not specified, + * search params are combined in a logical AND operation. + * Only rows that match *all* search param criteria will be returned. + * + * If `filter_or` is TRUE, multiple search params are combined in a logical OR operation. + * Results will include rows that match **any** of the search criteria. + * + * String search params use case-insensitive matching. + * String search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions. + * example="dan%" will match "danger" and "Danzig" but not "David" + * example="D_m%" will match "Damage" and "dump" + * + * Integer search params can accept a single value or a comma separated list of values. The multiple + * values will be combined under a logical OR operation - results will match at least one of + * the given values. + * + * Most search params can accept "IS NULL" and "NOT NULL" as special expressions to match + * or exclude (respectively) rows where the column is null. + * + * Boolean search params accept only "true" and "false" as values. + * + * + * Get a **single theme** by id with [Theme](#!/Theme/theme) + * + * **Note**: Custom themes needs to be enabled by Looker. Unless custom themes are enabled, only the automatically generated default theme can be used. Please contact your Account Manager or help.looker.com to update your license for this feature. + * + * + */ + @Override + public void searchThemes(SearchThemesReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/themes/search", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SearchThemesResponse.Builder responseBuilder = SearchThemesResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get the default theme + * + * Returns the active theme object set as the default. + * + * The **default** theme name can be set in the UI on the Admin|Theme UI page + * + * The optional `ts` parameter can specify a different timestamp than "now." If specified, it returns the default theme at the time indicated. + * + */ + @Override + public void defaultTheme(DefaultThemeReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/themes/default", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DefaultThemeResponse.Builder responseBuilder = DefaultThemeResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Set the global default theme by theme name + * + * Only Admin users can call this function. + * + * Only an active theme with no expiration (`end_at` not set) can be assigned as the default theme. As long as a theme has an active record with no expiration, it can be set as the default. + * + * [Create Theme](#!/Theme/create) has detailed information on rules for default and active themes + * + * Returns the new specified default theme object. + * + * **Note**: Custom themes needs to be enabled by Looker. Unless custom themes are enabled, only the automatically generated default theme can be used. Please contact your Account Manager or help.looker.com to update your license for this feature. + * + * + */ + @Override + public void setDefaultTheme(SetDefaultThemeReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.put("/themes/default", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SetDefaultThemeResponse.Builder responseBuilder = SetDefaultThemeResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get active themes + * + * Returns an array of active themes. + * + * If the `name` parameter is specified, it will return an array with one theme if it's active and found. + * + * The optional `ts` parameter can specify a different timestamp than "now." + * + * **Note**: Custom themes needs to be enabled by Looker. Unless custom themes are enabled, only the automatically generated default theme can be used. Please contact your Account Manager or help.looker.com to update your license for this feature. + * + * + * + */ + @Override + public void activeThemes(ActiveThemesReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/themes/active", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ActiveThemesResponse.Builder responseBuilder = ActiveThemesResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get the named theme if it's active. Otherwise, return the default theme + * + * The optional `ts` parameter can specify a different timestamp than "now." + * Note: API users with `show` ability can call this function + * + * **Note**: Custom themes needs to be enabled by Looker. Unless custom themes are enabled, only the automatically generated default theme can be used. Please contact your Account Manager or help.looker.com to update your license for this feature. + * + * + */ + @Override + public void themeOrDefault(ThemeOrDefaultReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/themes/theme_or_default", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ThemeOrDefaultResponse.Builder responseBuilder = ThemeOrDefaultResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Validate a theme with the specified information + * + * Validates all values set for the theme, returning any errors encountered, or 200 OK if valid + * + * See [Create Theme](#!/Theme/create_theme) for constraints + * + * **Note**: Custom themes needs to be enabled by Looker. Unless custom themes are enabled, only the automatically generated default theme can be used. Please contact your Account Manager or help.looker.com to update your license for this feature. + * + * + */ + @Override + public void validateTheme(ValidateThemeReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/themes/validate", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ValidateThemeResponse.Builder responseBuilder = ValidateThemeResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get a theme by ID + * + * Use this to retrieve a specific theme, whether or not it's currently active. + * + * **Note**: Custom themes needs to be enabled by Looker. Unless custom themes are enabled, only the automatically generated default theme can be used. Please contact your Account Manager or help.looker.com to update your license for this feature. + * + * + */ + @Override + public void theme(ThemeReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/themes/{theme_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ThemeResponse.Builder responseBuilder = ThemeResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update the theme by id. + * + * **Note**: Custom themes needs to be enabled by Looker. Unless custom themes are enabled, only the automatically generated default theme can be used. Please contact your Account Manager or help.looker.com to update your license for this feature. + * + * + */ + @Override + public void updateTheme(UpdateThemeReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/themes/{theme_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateThemeResponse.Builder responseBuilder = UpdateThemeResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Delete a specific theme by id + * + * This operation permanently deletes the identified theme from the database. + * + * Because multiple themes can have the same name (with different activation time spans) themes can only be deleted by ID. + * + * All IDs associated with a theme name can be retrieved by searching for the theme name with [Theme Search](#!/Theme/search). + * + * **Note**: Custom themes needs to be enabled by Looker. Unless custom themes are enabled, only the automatically generated default theme can be used. Please contact your Account Manager or help.looker.com to update your license for this feature. + * + * + */ + @Override + public void deleteTheme(DeleteThemeReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/themes/{theme_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteThemeResponse.Builder responseBuilder = DeleteThemeResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + //#endregion Theme: Manage Themes + + //#region User: Manage Users + + /** + * ### Search email credentials + * + * Returns all credentials_email records that match the given search criteria. + * + * If multiple search params are given and `filter_or` is FALSE or not specified, + * search params are combined in a logical AND operation. + * Only rows that match *all* search param criteria will be returned. + * + * If `filter_or` is TRUE, multiple search params are combined in a logical OR operation. + * Results will include rows that match **any** of the search criteria. + * + * String search params use case-insensitive matching. + * String search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions. + * example="dan%" will match "danger" and "Danzig" but not "David" + * example="D_m%" will match "Damage" and "dump" + * + * Integer search params can accept a single value or a comma separated list of values. The multiple + * values will be combined under a logical OR operation - results will match at least one of + * the given values. + * + * Most search params can accept "IS NULL" and "NOT NULL" as special expressions to match + * or exclude (respectively) rows where the column is null. + * + * Boolean search params accept only "true" and "false" as values. + * + * + */ + @Override + public void searchCredentialsEmail(SearchCredentialsEmailReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/credentials_email/search", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SearchCredentialsEmailResponse.Builder responseBuilder = SearchCredentialsEmailResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about the current user; i.e. the user account currently calling the API. + * + */ + @Override + public void me(MeReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/user", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + MeResponse.Builder responseBuilder = MeResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about all users. + * + */ + @Override + public void allUsers(AllUsersReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/users", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllUsersResponse.Builder responseBuilder = AllUsersResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create a user with the specified information. + * + */ + @Override + public void createUser(CreateUserReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/users", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateUserResponse.Builder responseBuilder = CreateUserResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Search users + * + * Returns all* user records that match the given search criteria. + * + * If multiple search params are given and `filter_or` is FALSE or not specified, + * search params are combined in a logical AND operation. + * Only rows that match *all* search param criteria will be returned. + * + * If `filter_or` is TRUE, multiple search params are combined in a logical OR operation. + * Results will include rows that match **any** of the search criteria. + * + * String search params use case-insensitive matching. + * String search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions. + * example="dan%" will match "danger" and "Danzig" but not "David" + * example="D_m%" will match "Damage" and "dump" + * + * Integer search params can accept a single value or a comma separated list of values. The multiple + * values will be combined under a logical OR operation - results will match at least one of + * the given values. + * + * Most search params can accept "IS NULL" and "NOT NULL" as special expressions to match + * or exclude (respectively) rows where the column is null. + * + * Boolean search params accept only "true" and "false" as values. + * + * + * (*) Results are always filtered to the level of information the caller is permitted to view. + * Looker admins can see all user details; normal users in an open system can see + * names of other users but no details; normal users in a closed system can only see + * names of other users who are members of the same group as the user. + * + * + */ + @Override + public void searchUsers(SearchUsersReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/users/search", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SearchUsersResponse.Builder responseBuilder = SearchUsersResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Search for user accounts by name + * + * Returns all user accounts where `first_name` OR `last_name` OR `email` field values match a pattern. + * The pattern can contain `%` and `_` wildcards as in SQL LIKE expressions. + * + * Any additional search params will be combined into a logical AND expression. + * + */ + @Override + public void searchUsersNames(SearchUsersNamesReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/users/search/names/{pattern}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SearchUsersNamesResponse.Builder responseBuilder = SearchUsersNamesResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about the user with a specific id. + * + * If the caller is an admin or the caller is the user being specified, then full user information will + * be returned. Otherwise, a minimal 'public' variant of the user information will be returned. This contains + * The user name and avatar url, but no sensitive information. + * + */ + @Override + public void user(UserReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/users/{user_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UserResponse.Builder responseBuilder = UserResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update information about the user with a specific id. + * + */ + @Override + public void updateUser(UpdateUserReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/users/{user_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateUserResponse.Builder responseBuilder = UpdateUserResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Delete the user with a specific id. + * + * **DANGER** this will delete the user and all looks and other information owned by the user. + * + */ + @Override + public void deleteUser(DeleteUserReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/users/{user_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteUserResponse.Builder responseBuilder = DeleteUserResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about the user with a credential of given type with specific id. + * + * This is used to do things like find users by their embed external_user_id. Or, find the user with + * a given api3 client_id, etc. The 'credential_type' matches the 'type' name of the various credential + * types. It must be one of the values listed in the table below. The 'credential_id' is your unique Id + * for the user and is specific to each type of credential. + * + * An example using the Ruby sdk might look like: + * + * `sdk.user_for_credential('embed', 'customer-4959425')` + * + * This table shows the supported 'Credential Type' strings. The right column is for reference; it shows + * which field in the given credential type is actually searched when finding a user with the supplied + * 'credential_id'. + * + * | Credential Types | Id Field Matched | + * | ---------------- | ---------------- | + * | email | email | + * | google | google_user_id | + * | saml | saml_user_id | + * | oidc | oidc_user_id | + * | ldap | ldap_id | + * | api | token | + * | api3 | client_id | + * | embed | external_user_id | + * | looker_openid | email | + * + * **NOTE**: The 'api' credential type was only used with the legacy Looker query API and is no longer supported. The credential type for API you are currently looking at is 'api3'. + * + * + */ + @Override + public void userForCredential(UserForCredentialReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/users/credential/{credential_type}/{credential_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UserForCredentialResponse.Builder responseBuilder = UserForCredentialResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Email/password login information for the specified user. + */ + @Override + public void userCredentialsEmail(UserCredentialsEmailReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/users/{user_id}/credentials_email", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UserCredentialsEmailResponse.Builder responseBuilder = UserCredentialsEmailResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Email/password login information for the specified user. + */ + @Override + public void createUserCredentialsEmail(CreateUserCredentialsEmailReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/users/{user_id}/credentials_email", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateUserCredentialsEmailResponse.Builder responseBuilder = CreateUserCredentialsEmailResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Email/password login information for the specified user. + */ + @Override + public void updateUserCredentialsEmail(UpdateUserCredentialsEmailReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/users/{user_id}/credentials_email", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateUserCredentialsEmailResponse.Builder responseBuilder = UpdateUserCredentialsEmailResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Email/password login information for the specified user. + */ + @Override + public void deleteUserCredentialsEmail(DeleteUserCredentialsEmailReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/users/{user_id}/credentials_email", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteUserCredentialsEmailResponse.Builder responseBuilder = DeleteUserCredentialsEmailResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Two-factor login information for the specified user. + */ + @Override + public void userCredentialsTotp(UserCredentialsTotpReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/users/{user_id}/credentials_totp", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UserCredentialsTotpResponse.Builder responseBuilder = UserCredentialsTotpResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Two-factor login information for the specified user. + */ + @Override + public void createUserCredentialsTotp(CreateUserCredentialsTotpReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/users/{user_id}/credentials_totp", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateUserCredentialsTotpResponse.Builder responseBuilder = CreateUserCredentialsTotpResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Two-factor login information for the specified user. + */ + @Override + public void deleteUserCredentialsTotp(DeleteUserCredentialsTotpReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/users/{user_id}/credentials_totp", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteUserCredentialsTotpResponse.Builder responseBuilder = DeleteUserCredentialsTotpResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### LDAP login information for the specified user. + */ + @Override + public void userCredentialsLdap(UserCredentialsLdapReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/users/{user_id}/credentials_ldap", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UserCredentialsLdapResponse.Builder responseBuilder = UserCredentialsLdapResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### LDAP login information for the specified user. + */ + @Override + public void deleteUserCredentialsLdap(DeleteUserCredentialsLdapReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/users/{user_id}/credentials_ldap", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteUserCredentialsLdapResponse.Builder responseBuilder = DeleteUserCredentialsLdapResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Google authentication login information for the specified user. + */ + @Override + public void userCredentialsGoogle(UserCredentialsGoogleReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/users/{user_id}/credentials_google", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UserCredentialsGoogleResponse.Builder responseBuilder = UserCredentialsGoogleResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Google authentication login information for the specified user. + */ + @Override + public void deleteUserCredentialsGoogle(DeleteUserCredentialsGoogleReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/users/{user_id}/credentials_google", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteUserCredentialsGoogleResponse.Builder responseBuilder = DeleteUserCredentialsGoogleResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Saml authentication login information for the specified user. + */ + @Override + public void userCredentialsSaml(UserCredentialsSamlReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/users/{user_id}/credentials_saml", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UserCredentialsSamlResponse.Builder responseBuilder = UserCredentialsSamlResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Saml authentication login information for the specified user. + */ + @Override + public void deleteUserCredentialsSaml(DeleteUserCredentialsSamlReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/users/{user_id}/credentials_saml", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteUserCredentialsSamlResponse.Builder responseBuilder = DeleteUserCredentialsSamlResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### OpenID Connect (OIDC) authentication login information for the specified user. + */ + @Override + public void userCredentialsOidc(UserCredentialsOidcReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/users/{user_id}/credentials_oidc", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UserCredentialsOidcResponse.Builder responseBuilder = UserCredentialsOidcResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### OpenID Connect (OIDC) authentication login information for the specified user. + */ + @Override + public void deleteUserCredentialsOidc(DeleteUserCredentialsOidcReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/users/{user_id}/credentials_oidc", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteUserCredentialsOidcResponse.Builder responseBuilder = DeleteUserCredentialsOidcResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### API 3 login information for the specified user. This is for the newer API keys that can be added for any user. + */ + @Override + public void userCredentialsApi3(UserCredentialsApi3Req request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/users/{user_id}/credentials_api3/{credentials_api3_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UserCredentialsApi3Response.Builder responseBuilder = UserCredentialsApi3Response.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### API 3 login information for the specified user. This is for the newer API keys that can be added for any user. + */ + @Override + public void deleteUserCredentialsApi3(DeleteUserCredentialsApi3Req request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/users/{user_id}/credentials_api3/{credentials_api3_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteUserCredentialsApi3Response.Builder responseBuilder = DeleteUserCredentialsApi3Response.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### API 3 login information for the specified user. This is for the newer API keys that can be added for any user. + */ + @Override + public void allUserCredentialsApi3s(AllUserCredentialsApi3sReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/users/{user_id}/credentials_api3", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllUserCredentialsApi3sResponse.Builder responseBuilder = AllUserCredentialsApi3sResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### API 3 login information for the specified user. This is for the newer API keys that can be added for any user. + */ + @Override + public void createUserCredentialsApi3(CreateUserCredentialsApi3Req request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/users/{user_id}/credentials_api3", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateUserCredentialsApi3Response.Builder responseBuilder = CreateUserCredentialsApi3Response.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Embed login information for the specified user. + */ + @Override + public void userCredentialsEmbed(UserCredentialsEmbedReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/users/{user_id}/credentials_embed/{credentials_embed_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UserCredentialsEmbedResponse.Builder responseBuilder = UserCredentialsEmbedResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Embed login information for the specified user. + */ + @Override + public void deleteUserCredentialsEmbed(DeleteUserCredentialsEmbedReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/users/{user_id}/credentials_embed/{credentials_embed_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteUserCredentialsEmbedResponse.Builder responseBuilder = DeleteUserCredentialsEmbedResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Embed login information for the specified user. + */ + @Override + public void allUserCredentialsEmbeds(AllUserCredentialsEmbedsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/users/{user_id}/credentials_embed", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllUserCredentialsEmbedsResponse.Builder responseBuilder = AllUserCredentialsEmbedsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Looker Openid login information for the specified user. Used by Looker Analysts. + */ + @Override + public void userCredentialsLookerOpenid(UserCredentialsLookerOpenidReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/users/{user_id}/credentials_looker_openid", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UserCredentialsLookerOpenidResponse.Builder responseBuilder = UserCredentialsLookerOpenidResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Looker Openid login information for the specified user. Used by Looker Analysts. + */ + @Override + public void deleteUserCredentialsLookerOpenid(DeleteUserCredentialsLookerOpenidReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/users/{user_id}/credentials_looker_openid", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteUserCredentialsLookerOpenidResponse.Builder responseBuilder = DeleteUserCredentialsLookerOpenidResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Web login session for the specified user. + */ + @Override + public void userSession(UserSessionReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/users/{user_id}/sessions/{session_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UserSessionResponse.Builder responseBuilder = UserSessionResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Web login session for the specified user. + */ + @Override + public void deleteUserSession(DeleteUserSessionReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/users/{user_id}/sessions/{session_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteUserSessionResponse.Builder responseBuilder = DeleteUserSessionResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Web login session for the specified user. + */ + @Override + public void allUserSessions(AllUserSessionsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/users/{user_id}/sessions", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllUserSessionsResponse.Builder responseBuilder = AllUserSessionsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create a password reset token. + * This will create a cryptographically secure random password reset token for the user. + * If the user already has a password reset token then this invalidates the old token and creates a new one. + * The token is expressed as the 'password_reset_url' of the user's email/password credential object. + * This takes an optional 'expires' param to indicate if the new token should be an expiring token. + * Tokens that expire are typically used for self-service password resets for existing users. + * Invitation emails for new users typically are not set to expire. + * The expire period is always 60 minutes when expires is enabled. + * This method can be called with an empty body. + * + */ + @Override + public void createUserCredentialsEmailPasswordReset(CreateUserCredentialsEmailPasswordResetReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/users/{user_id}/credentials_email/password_reset", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateUserCredentialsEmailPasswordResetResponse.Builder responseBuilder = CreateUserCredentialsEmailPasswordResetResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about roles of a given user + * + */ + @Override + public void userRoles(UserRolesReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/users/{user_id}/roles", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UserRolesResponse.Builder responseBuilder = UserRolesResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Set roles of the user with a specific id. + * + */ + @Override + public void setUserRoles(SetUserRolesReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.put("/users/{user_id}/roles", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SetUserRolesResponse.Builder responseBuilder = SetUserRolesResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get user attribute values for a given user. + * + * Returns the values of specified user attributes (or all user attributes) for a certain user. + * + * A value for each user attribute is searched for in the following locations, in this order: + * + * 1. in the user's account information + * 1. in groups that the user is a member of + * 1. the default value of the user attribute + * + * If more than one group has a value defined for a user attribute, the group with the lowest rank wins. + * + * The response will only include user attributes for which values were found. Use `include_unset=true` to include + * empty records for user attributes with no value. + * + * The value of all hidden user attributes will be blank. + * + */ + @Override + public void userAttributeUserValues(UserAttributeUserValuesReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/users/{user_id}/attribute_values", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UserAttributeUserValuesResponse.Builder responseBuilder = UserAttributeUserValuesResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Store a custom value for a user attribute in a user's account settings. + * + * Per-user user attribute values take precedence over group or default values. + * + */ + @Override + public void setUserAttributeUserValue(SetUserAttributeUserValueReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/users/{user_id}/attribute_values/{user_attribute_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SetUserAttributeUserValueResponse.Builder responseBuilder = SetUserAttributeUserValueResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Delete a user attribute value from a user's account settings. + * + * After the user attribute value is deleted from the user's account settings, subsequent requests + * for the user attribute value for this user will draw from the user's groups or the default + * value of the user attribute. See [Get User Attribute Values](#!/User/user_attribute_user_values) for more + * information about how user attribute values are resolved. + * + */ + @Override + public void deleteUserAttributeUserValue(DeleteUserAttributeUserValueReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/users/{user_id}/attribute_values/{user_attribute_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteUserAttributeUserValueResponse.Builder responseBuilder = DeleteUserAttributeUserValueResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Send a password reset token. + * This will send a password reset email to the user. If a password reset token does not already exist + * for this user, it will create one and then send it. + * If the user has not yet set up their account, it will send a setup email to the user. + * The URL sent in the email is expressed as the 'password_reset_url' of the user's email/password credential object. + * Password reset URLs will expire in 60 minutes. + * This method can be called with an empty body. + * + */ + @Override + public void sendUserCredentialsEmailPasswordReset(SendUserCredentialsEmailPasswordResetReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/users/{user_id}/credentials_email/send_password_reset", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SendUserCredentialsEmailPasswordResetResponse.Builder responseBuilder = SendUserCredentialsEmailPasswordResetResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Change a disabled user's email addresses + * + * Allows the admin to change the email addresses for all the user's + * associated credentials. Will overwrite all associated email addresses with + * the value supplied in the 'email' body param. + * The user's 'is_disabled' status must be true. + * + */ + @Override + public void wipeoutUserEmails(WipeoutUserEmailsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/users/{user_id}/update_emails", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + WipeoutUserEmailsResponse.Builder responseBuilder = WipeoutUserEmailsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * Create an embed user from an external user ID + * + */ + @Override + public void createEmbedUser(CreateEmbedUserReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/users/embed_user", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateEmbedUserResponse.Builder responseBuilder = CreateEmbedUserResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + //#endregion User: Manage Users + + //#region UserAttribute: Manage User Attributes + + /** + * ### Get information about all user attributes. + * + */ + @Override + public void allUserAttributes(AllUserAttributesReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/user_attributes", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllUserAttributesResponse.Builder responseBuilder = AllUserAttributesResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create a new user attribute + * + * Permission information for a user attribute is conveyed through the `can` and `user_can_edit` fields. + * The `user_can_edit` field indicates whether an attribute is user-editable _anywhere_ in the application. + * The `can` field gives more granular access information, with the `set_value` child field indicating whether + * an attribute's value can be set by [Setting the User Attribute User Value](#!/User/set_user_attribute_user_value). + * + * Note: `name` and `label` fields must be unique across all user attributes in the Looker instance. + * Attempting to create a new user attribute with a name or label that duplicates an existing + * user attribute will fail with a 422 error. + * + */ + @Override + public void createUserAttribute(CreateUserAttributeReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/user_attributes", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateUserAttributeResponse.Builder responseBuilder = CreateUserAttributeResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about a user attribute. + * + */ + @Override + public void userAttribute(UserAttributeReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/user_attributes/{user_attribute_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UserAttributeResponse.Builder responseBuilder = UserAttributeResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update a user attribute definition. + * + */ + @Override + public void updateUserAttribute(UpdateUserAttributeReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/user_attributes/{user_attribute_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateUserAttributeResponse.Builder responseBuilder = UpdateUserAttributeResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Delete a user attribute (admin only). + * + */ + @Override + public void deleteUserAttribute(DeleteUserAttributeReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/user_attributes/{user_attribute_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteUserAttributeResponse.Builder responseBuilder = DeleteUserAttributeResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Returns all values of a user attribute defined by user groups, in precedence order. + * + * A user may be a member of multiple groups which define different values for a given user attribute. + * The order of group-values in the response determines precedence for selecting which group-value applies + * to a given user. For more information, see [Set User Attribute Group Values](#!/UserAttribute/set_user_attribute_group_values). + * + * Results will only include groups that the caller's user account has permission to see. + * + */ + @Override + public void allUserAttributeGroupValues(AllUserAttributeGroupValuesReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/user_attributes/{user_attribute_id}/group_values", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllUserAttributeGroupValuesResponse.Builder responseBuilder = AllUserAttributeGroupValuesResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Define values for a user attribute across a set of groups, in priority order. + * + * This function defines all values for a user attribute defined by user groups. This is a global setting, potentially affecting + * all users in the system. This function replaces any existing group value definitions for the indicated user attribute. + * + * The value of a user attribute for a given user is determined by searching the following locations, in this order: + * + * 1. the user's account settings + * 2. the groups that the user is a member of + * 3. the default value of the user attribute, if any + * + * The user may be a member of multiple groups which define different values for that user attribute. The order of items in the group_values parameter + * determines which group takes priority for that user. Lowest array index wins. + * + * An alternate method to indicate the selection precedence of group-values is to assign numbers to the 'rank' property of each + * group-value object in the array. Lowest 'rank' value wins. If you use this technique, you must assign a + * rank value to every group-value object in the array. + * + * To set a user attribute value for a single user, see [Set User Attribute User Value](#!/User/set_user_attribute_user_value). + * To set a user attribute value for all members of a group, see [Set User Attribute Group Value](#!/Group/update_user_attribute_group_value). + * + */ + @Override + public void setUserAttributeGroupValues(SetUserAttributeGroupValuesReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/user_attributes/{user_attribute_id}/group_values", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SetUserAttributeGroupValuesResponse.Builder responseBuilder = SetUserAttributeGroupValuesResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + //#endregion UserAttribute: Manage User Attributes + + //#region Workspace: Manage Workspaces + + /** + * ### Get All Workspaces + * + * Returns all workspaces available to the calling user. + * + */ + @Override + public void allWorkspaces(AllWorkspacesReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/workspaces", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllWorkspacesResponse.Builder responseBuilder = AllWorkspacesResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get A Workspace + * + * Returns information about a workspace such as the git status and selected branches + * of all projects available to the caller's user account. + * + * A workspace defines which versions of project files will be used to evaluate expressions + * and operations that use model definitions - operations such as running queries or rendering dashboards. + * Each project has its own git repository, and each project in a workspace may be configured to reference + * particular branch or revision within their respective repositories. + * + * There are two predefined workspaces available: "production" and "dev". + * + * The production workspace is shared across all Looker users. Models in the production workspace are read-only. + * Changing files in production is accomplished by modifying files in a git branch and using Pull Requests + * to merge the changes from the dev branch into the production branch, and then telling + * Looker to sync with production. + * + * The dev workspace is local to each Looker user. Changes made to project/model files in the dev workspace only affect + * that user, and only when the dev workspace is selected as the active workspace for the API session. + * (See set_session_workspace()). + * + * The dev workspace is NOT unique to an API session. Two applications accessing the Looker API using + * the same user account will see the same files in the dev workspace. To avoid collisions between + * API clients it's best to have each client login with API3 credentials for a different user account. + * + * Changes made to files in a dev workspace are persistent across API sessions. It's a good + * idea to commit any changes you've made to the git repository, but not strictly required. Your modified files + * reside in a special user-specific directory on the Looker server and will still be there when you login in again + * later and use update_session(workspace_id: "dev") to select the dev workspace for the new API session. + * + */ + @Override + public void workspace(WorkspaceReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/workspaces/{workspace_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + WorkspaceResponse.Builder responseBuilder = WorkspaceResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + //#endregion Workspace: Manage Workspaces +} \ No newline at end of file diff --git a/proto/grpc_proxy/src/main/java/com/google/looker/server/sdk/LookerStreamingServiceImpl.java b/proto/grpc_proxy/src/main/java/com/google/looker/server/sdk/LookerStreamingServiceImpl.java new file mode 100644 index 000000000..39c01fc75 --- /dev/null +++ b/proto/grpc_proxy/src/main/java/com/google/looker/server/sdk/LookerStreamingServiceImpl.java @@ -0,0 +1,16375 @@ +// MIT License +// +// Copyright (c) 2021 Looker Data Sciences, Inc. +// +// Permission is hereby granted, free of charge, to any person obtaining a copy +// of this software and associated documentation files (the "Software"), to deal +// in the Software without restriction, including without limitation the rights +// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +// copies of the Software, and to permit persons to whom the Software is +// furnished to do so, subject to the following conditions: +// +// The above copyright notice and this permission notice shall be included in all +// copies or substantial portions of the Software. +// +// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +// SOFTWARE. +// + +// 438 API methods + + +package com.google.looker.server.sdk; + +import com.google.looker.grpc.services.*; +import com.google.looker.grpc.services.LookerStreamingServiceGrpc.LookerStreamingServiceImplBase; +import com.google.looker.server.rtl.LookerClient; +import com.google.looker.server.rtl.LookerClientResponse; +import com.google.protobuf.InvalidProtocolBufferException; +import com.google.protobuf.util.JsonFormat; +import io.grpc.Status; +import io.grpc.stub.StreamObserver; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +public class LookerStreamingServiceImpl extends LookerStreamingServiceImplBase { + + final private static Logger LOGGER = LoggerFactory.getLogger(LookerStreamingServiceImpl.class); + + final private LookerClient lookerClient; + + public LookerStreamingServiceImpl() { + lookerClient = new LookerClient("4.0"); + } + + + //#region Alert: Alert + + /** + * ### Search Alerts + * + */ + @Override + public void searchAlerts(SearchAlertsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/alerts/search", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SearchAlertsResponse.Builder responseBuilder = SearchAlertsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + SearchAlertsStreamResponse.Builder responseBuilder2 = SearchAlertsStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get an alert by a given alert ID + * + */ + @Override + public void getAlert(GetAlertReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/alerts/{alert_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + GetAlertResponse.Builder responseBuilder = GetAlertResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update an alert + * # Required fields: `owner_id`, `field`, `destinations`, `comparison_type`, `threshold`, `cron` + * # + * + */ + @Override + public void updateAlert(UpdateAlertReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.put("/alerts/{alert_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateAlertResponse.Builder responseBuilder = UpdateAlertResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update select alert fields + * # Available fields: `owner_id`, `is_disabled`, `disabled_reason`, `is_public`, `threshold` + * # + * + */ + @Override + public void updateAlertField(UpdateAlertFieldReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/alerts/{alert_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateAlertFieldResponse.Builder responseBuilder = UpdateAlertFieldResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Delete an alert by a given alert ID + * + */ + @Override + public void deleteAlert(DeleteAlertReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/alerts/{alert_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteAlertResponse.Builder responseBuilder = DeleteAlertResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create a new alert and return details of the newly created object + * + * Required fields: `field`, `destinations`, `comparison_type`, `threshold`, `cron` + * + * Example Request: + * Run alert on dashboard element '103' at 5am every day. Send an email to 'test@test.com' if inventory for Los Angeles (using dashboard filter `Warehouse Name`) is lower than 1,000 + * ``` + * { + * "cron": "0 5 * * *", + * "custom_title": "Alert when LA inventory is low", + * "dashboard_element_id": 103, + * "applied_dashboard_filters": [ + * { + * "filter_title": "Warehouse Name", + * "field_name": "distribution_centers.name", + * "filter_value": "Los Angeles CA", + * "filter_description": "is Los Angeles CA" + * } + * ], + * "comparison_type": "LESS_THAN", + * "destinations": [ + * { + * "destination_type": "EMAIL", + * "email_address": "test@test.com" + * } + * ], + * "field": { + * "title": "Number on Hand", + * "name": "inventory_items.number_on_hand" + * }, + * "is_disabled": false, + * "is_public": true, + * "threshold": 1000 + * } + * ``` + * + */ + @Override + public void createAlert(CreateAlertReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/alerts", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateAlertResponse.Builder responseBuilder = CreateAlertResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Enqueue an Alert by ID + * + */ + @Override + public void enqueueAlert(EnqueueAlertReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/alerts/{alert_id}/enqueue", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + EnqueueAlertResponse.Builder responseBuilder = EnqueueAlertResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + //#endregion Alert: Alert + + //#region ApiAuth: API Authentication + + /** + * ### Present client credentials to obtain an authorization token + * + * Looker API implements the OAuth2 [Resource Owner Password Credentials Grant](https://looker.com/docs/r/api/outh2_resource_owner_pc) pattern. + * The client credentials required for this login must be obtained by creating an API3 key on a user account + * in the Looker Admin console. The API3 key consists of a public `client_id` and a private `client_secret`. + * + * The access token returned by `login` must be used in the HTTP Authorization header of subsequent + * API requests, like this: + * ``` + * Authorization: token 4QDkCyCtZzYgj4C2p2cj3csJH7zqS5RzKs2kTnG4 + * ``` + * Replace "4QDkCy..." with the `access_token` value returned by `login`. + * The word `token` is a string literal and must be included exactly as shown. + * + * This function can accept `client_id` and `client_secret` parameters as URL query params or as www-form-urlencoded params in the body of the HTTP request. Since there is a small risk that URL parameters may be visible to intermediate nodes on the network route (proxies, routers, etc), passing credentials in the body of the request is considered more secure than URL params. + * + * Example of passing credentials in the HTTP request body: + * ```` + * POST HTTP /login + * Content-Type: application/x-www-form-urlencoded + * + * client_id=CGc9B7v7J48dQSJvxxx&client_secret=nNVS9cSS3xNpSC9JdsBvvvvv + * ```` + * + * ### Best Practice: + * Always pass credentials in body params. Pass credentials in URL query params **only** when you cannot pass body params due to application, tool, or other limitations. + * + * For more information and detailed examples of Looker API authorization, see [How to Authenticate to Looker API3](https://github.com/looker/looker-sdk-ruby/blob/master/authentication.md). + * + */ + @Override + public void login(LoginReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/login", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + LoginResponse.Builder responseBuilder = LoginResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create an access token that runs as a given user. + * + * This can only be called by an authenticated admin user. It allows that admin to generate a new + * authentication token for the user with the given user id. That token can then be used for subsequent + * API calls - which are then performed *as* that target user. + * + * The target user does *not* need to have a pre-existing API client_id/client_secret pair. And, no such + * credentials are created by this call. + * + * This allows for building systems where api user authentication for an arbitrary number of users is done + * outside of Looker and funneled through a single 'service account' with admin permissions. Note that a + * new access token is generated on each call. If target users are going to be making numerous API + * calls in a short period then it is wise to cache this authentication token rather than call this before + * each of those API calls. + * + * See 'login' for more detail on the access token and how to use it. + * + */ + @Override + public void loginUser(LoginUserReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/login/{user_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + LoginUserResponse.Builder responseBuilder = LoginUserResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Logout of the API and invalidate the current access token. + * + */ + @Override + public void logout(LogoutReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/logout", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + LogoutResponse.Builder responseBuilder = LogoutResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + //#endregion ApiAuth: API Authentication + + //#region Auth: Manage User Authentication Configuration + + /** + * ### Create an embed secret using the specified information. + * + * The value of the `secret` field will be set by Looker and returned. + * + */ + @Override + public void createEmbedSecret(CreateEmbedSecretReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/embed_config/secrets", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateEmbedSecretResponse.Builder responseBuilder = CreateEmbedSecretResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Delete an embed secret. + * + */ + @Override + public void deleteEmbedSecret(DeleteEmbedSecretReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/embed_config/secrets/{embed_secret_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteEmbedSecretResponse.Builder responseBuilder = DeleteEmbedSecretResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create SSO Embed URL + * + * Creates an SSO embed URL and cryptographically signs it with an embed secret. + * This signed URL can then be used to instantiate a Looker embed session in a PBL web application. + * Do not make any modifications to this URL - any change may invalidate the signature and + * cause the URL to fail to load a Looker embed session. + * + * A signed SSO embed URL can only be used once. After it has been used to request a page from the + * Looker server, the URL is invalid. Future requests using the same URL will fail. This is to prevent + * 'replay attacks'. + * + * The `target_url` property must be a complete URL of a Looker UI page - scheme, hostname, path and query params. + * To load a dashboard with id 56 and with a filter of `Date=1 years`, the looker URL would look like `https:/myname.looker.com/dashboards/56?Date=1%20years`. + * The best way to obtain this target_url is to navigate to the desired Looker page in your web browser, + * copy the URL shown in the browser address bar and paste it into the `target_url` property as a quoted string value in this API request. + * + * Permissions for the embed user are defined by the groups in which the embed user is a member (group_ids property) + * and the lists of models and permissions assigned to the embed user. + * At a minimum, you must provide values for either the group_ids property, or both the models and permissions properties. + * These properties are additive; an embed user can be a member of certain groups AND be granted access to models and permissions. + * + * The embed user's access is the union of permissions granted by the group_ids, models, and permissions properties. + * + * This function does not strictly require all group_ids, user attribute names, or model names to exist at the moment the + * SSO embed url is created. Unknown group_id, user attribute names or model names will be passed through to the output URL. + * To diagnose potential problems with an SSO embed URL, you can copy the signed URL into the Embed URI Validator text box in `/admin/embed`. + * + * The `secret_id` parameter is optional. If specified, its value must be the id of an active secret defined in the Looker instance. + * if not specified, the URL will be signed using the newest active secret defined in the Looker instance. + * + * #### Security Note + * Protect this signed URL as you would an access token or password credentials - do not write + * it to disk, do not pass it to a third party, and only pass it through a secure HTTPS + * encrypted transport. + * + */ + @Override + public void createSsoEmbedUrl(CreateSsoEmbedUrlReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/embed/sso_url", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateSsoEmbedUrlResponse.Builder responseBuilder = CreateSsoEmbedUrlResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create an Embed URL + * + * Creates an embed URL that runs as the Looker user making this API call. ("Embed as me") + * This embed URL can then be used to instantiate a Looker embed session in a + * "Powered by Looker" (PBL) web application. + * + * This is similar to Private Embedding (https://docs.looker.com/r/admin/embed/private-embed). Instead of + * of logging into the Web UI to authenticate, the user has already authenticated against the API to be able to + * make this call. However, unlike Private Embed where the user has access to any other part of the Looker UI, + * the embed web session created by requesting the EmbedUrlResponse.url in a browser only has access to + * content visible under the `/embed` context. + * + * An embed URL can only be used once, and must be used within 5 minutes of being created. After it + * has been used to request a page from the Looker server, the URL is invalid. Future requests using + * the same URL will fail. This is to prevent 'replay attacks'. + * + * The `target_url` property must be a complete URL of a Looker Embedded UI page - scheme, hostname, path starting with "/embed" and query params. + * To load a dashboard with id 56 and with a filter of `Date=1 years`, the looker Embed URL would look like `https://myname.looker.com/embed/dashboards/56?Date=1%20years`. + * The best way to obtain this target_url is to navigate to the desired Looker page in your web browser, + * copy the URL shown in the browser address bar, insert "/embed" after the host/port, and paste it into the `target_url` property as a quoted string value in this API request. + * + * #### Security Note + * Protect this embed URL as you would an access token or password credentials - do not write + * it to disk, do not pass it to a third party, and only pass it through a secure HTTPS + * encrypted transport. + * + */ + @Override + public void createEmbedUrlAsMe(CreateEmbedUrlAsMeReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/embed/token_url/me", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateEmbedUrlAsMeResponse.Builder responseBuilder = CreateEmbedUrlAsMeResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get the LDAP configuration. + * + * Looker can be optionally configured to authenticate users against an Active Directory or other LDAP directory server. + * LDAP setup requires coordination with an administrator of that directory server. + * + * Only Looker administrators can read and update the LDAP configuration. + * + * Configuring LDAP impacts authentication for all users. This configuration should be done carefully. + * + * Looker maintains a single LDAP configuration. It can be read and updated. Updates only succeed if the new state will be valid (in the sense that all required fields are populated); it is up to you to ensure that the configuration is appropriate and correct). + * + * LDAP is enabled or disabled for Looker using the **enabled** field. + * + * Looker will never return an **auth_password** field. That value can be set, but never retrieved. + * + * See the [Looker LDAP docs](https://www.looker.com/docs/r/api/ldap_setup) for additional information. + * + */ + @Override + public void ldapConfig(LdapConfigReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/ldap_config", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + LdapConfigResponse.Builder responseBuilder = LdapConfigResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update the LDAP configuration. + * + * Configuring LDAP impacts authentication for all users. This configuration should be done carefully. + * + * Only Looker administrators can read and update the LDAP configuration. + * + * LDAP is enabled or disabled for Looker using the **enabled** field. + * + * It is **highly** recommended that any LDAP setting changes be tested using the APIs below before being set globally. + * + * See the [Looker LDAP docs](https://www.looker.com/docs/r/api/ldap_setup) for additional information. + * + */ + @Override + public void updateLdapConfig(UpdateLdapConfigReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/ldap_config", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateLdapConfigResponse.Builder responseBuilder = UpdateLdapConfigResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Test the connection settings for an LDAP configuration. + * + * This tests that the connection is possible given a connection_host and connection_port. + * + * **connection_host** and **connection_port** are required. **connection_tls** is optional. + * + * Example: + * ```json + * { + * "connection_host": "ldap.example.com", + * "connection_port": "636", + * "connection_tls": true + * } + * ``` + * + * No authentication to the LDAP server is attempted. + * + * The active LDAP settings are not modified. + * + */ + @Override + public void testLdapConfigConnection(TestLdapConfigConnectionReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.put("/ldap_config/test_connection", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + TestLdapConfigConnectionResponse.Builder responseBuilder = TestLdapConfigConnectionResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Test the connection authentication settings for an LDAP configuration. + * + * This tests that the connection is possible and that a 'server' account to be used by Looker can authenticate to the LDAP server given connection and authentication information. + * + * **connection_host**, **connection_port**, and **auth_username**, are required. **connection_tls** and **auth_password** are optional. + * + * Example: + * ```json + * { + * "connection_host": "ldap.example.com", + * "connection_port": "636", + * "connection_tls": true, + * "auth_username": "cn=looker,dc=example,dc=com", + * "auth_password": "secret" + * } + * ``` + * + * Looker will never return an **auth_password**. If this request omits the **auth_password** field, then the **auth_password** value from the active config (if present) will be used for the test. + * + * The active LDAP settings are not modified. + * + * + */ + @Override + public void testLdapConfigAuth(TestLdapConfigAuthReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.put("/ldap_config/test_auth", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + TestLdapConfigAuthResponse.Builder responseBuilder = TestLdapConfigAuthResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Test the user authentication settings for an LDAP configuration without authenticating the user. + * + * This test will let you easily test the mapping for user properties and roles for any user without needing to authenticate as that user. + * + * This test accepts a full LDAP configuration along with a username and attempts to find the full info for the user from the LDAP server without actually authenticating the user. So, user password is not required.The configuration is validated before attempting to contact the server. + * + * **test_ldap_user** is required. + * + * The active LDAP settings are not modified. + * + * + */ + @Override + public void testLdapConfigUserInfo(TestLdapConfigUserInfoReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.put("/ldap_config/test_user_info", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + TestLdapConfigUserInfoResponse.Builder responseBuilder = TestLdapConfigUserInfoResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Test the user authentication settings for an LDAP configuration. + * + * This test accepts a full LDAP configuration along with a username/password pair and attempts to authenticate the user with the LDAP server. The configuration is validated before attempting the authentication. + * + * Looker will never return an **auth_password**. If this request omits the **auth_password** field, then the **auth_password** value from the active config (if present) will be used for the test. + * + * **test_ldap_user** and **test_ldap_password** are required. + * + * The active LDAP settings are not modified. + * + * + */ + @Override + public void testLdapConfigUserAuth(TestLdapConfigUserAuthReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.put("/ldap_config/test_user_auth", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + TestLdapConfigUserAuthResponse.Builder responseBuilder = TestLdapConfigUserAuthResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### List All OAuth Client Apps + * + * Lists all applications registered to use OAuth2 login with this Looker instance, including + * enabled and disabled apps. + * + * Results are filtered to include only the apps that the caller (current user) + * has permission to see. + * + */ + @Override + public void allOauthClientApps(AllOauthClientAppsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/oauth_client_apps", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllOauthClientAppsResponse.Builder responseBuilder = AllOauthClientAppsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + AllOauthClientAppsStreamResponse.Builder responseBuilder2 = AllOauthClientAppsStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get Oauth Client App + * + * Returns the registered app client with matching client_guid. + * + */ + @Override + public void oauthClientApp(OauthClientAppReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/oauth_client_apps/{client_guid}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + OauthClientAppResponse.Builder responseBuilder = OauthClientAppResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Register an OAuth2 Client App + * + * Registers details identifying an external web app or native app as an OAuth2 login client of the Looker instance. + * The app registration must provide a unique client_guid and redirect_uri that the app will present + * in OAuth login requests. If the client_guid and redirect_uri parameters in the login request do not match + * the app details registered with the Looker instance, the request is assumed to be a forgery and is rejected. + * + */ + @Override + public void registerOauthClientApp(RegisterOauthClientAppReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/oauth_client_apps/{client_guid}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + RegisterOauthClientAppResponse.Builder responseBuilder = RegisterOauthClientAppResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update OAuth2 Client App Details + * + * Modifies the details a previously registered OAuth2 login client app. + * + */ + @Override + public void updateOauthClientApp(UpdateOauthClientAppReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/oauth_client_apps/{client_guid}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateOauthClientAppResponse.Builder responseBuilder = UpdateOauthClientAppResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Delete OAuth Client App + * + * Deletes the registration info of the app with the matching client_guid. + * All active sessions and tokens issued for this app will immediately become invalid. + * + * ### Note: this deletion cannot be undone. + * + */ + @Override + public void deleteOauthClientApp(DeleteOauthClientAppReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/oauth_client_apps/{client_guid}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteOauthClientAppResponse.Builder responseBuilder = DeleteOauthClientAppResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Invalidate All Issued Tokens + * + * Immediately invalidates all auth codes, sessions, access tokens and refresh tokens issued for + * this app for ALL USERS of this app. + * + */ + @Override + public void invalidateTokens(InvalidateTokensReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/oauth_client_apps/{client_guid}/tokens", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + InvalidateTokensResponse.Builder responseBuilder = InvalidateTokensResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Activate an app for a user + * + * Activates a user for a given oauth client app. This indicates the user has been informed that + * the app will have access to the user's looker data, and that the user has accepted and allowed + * the app to use their Looker account. + * + * Activating a user for an app that the user is already activated with returns a success response. + * + */ + @Override + public void activateAppUser(ActivateAppUserReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/oauth_client_apps/{client_guid}/users/{user_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ActivateAppUserResponse.Builder responseBuilder = ActivateAppUserResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Deactivate an app for a user + * + * Deactivate a user for a given oauth client app. All tokens issued to the app for + * this user will be invalid immediately. Before the user can use the app with their + * Looker account, the user will have to read and accept an account use disclosure statement for the app. + * + * Admin users can deactivate other users, but non-admin users can only deactivate themselves. + * + * As with most REST DELETE operations, this endpoint does not return an error if the indicated + * resource (app or user) does not exist or has already been deactivated. + * + */ + @Override + public void deactivateAppUser(DeactivateAppUserReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/oauth_client_apps/{client_guid}/users/{user_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeactivateAppUserResponse.Builder responseBuilder = DeactivateAppUserResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get the OIDC configuration. + * + * Looker can be optionally configured to authenticate users against an OpenID Connect (OIDC) + * authentication server. OIDC setup requires coordination with an administrator of that server. + * + * Only Looker administrators can read and update the OIDC configuration. + * + * Configuring OIDC impacts authentication for all users. This configuration should be done carefully. + * + * Looker maintains a single OIDC configuation. It can be read and updated. Updates only succeed if the new state will be valid (in the sense that all required fields are populated); it is up to you to ensure that the configuration is appropriate and correct). + * + * OIDC is enabled or disabled for Looker using the **enabled** field. + * + */ + @Override + public void oidcConfig(OidcConfigReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/oidc_config", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + OidcConfigResponse.Builder responseBuilder = OidcConfigResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update the OIDC configuration. + * + * Configuring OIDC impacts authentication for all users. This configuration should be done carefully. + * + * Only Looker administrators can read and update the OIDC configuration. + * + * OIDC is enabled or disabled for Looker using the **enabled** field. + * + * It is **highly** recommended that any OIDC setting changes be tested using the APIs below before being set globally. + * + */ + @Override + public void updateOidcConfig(UpdateOidcConfigReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/oidc_config", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateOidcConfigResponse.Builder responseBuilder = UpdateOidcConfigResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get a OIDC test configuration by test_slug. + * + */ + @Override + public void oidcTestConfig(OidcTestConfigReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/oidc_test_configs/{test_slug}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + OidcTestConfigResponse.Builder responseBuilder = OidcTestConfigResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Delete a OIDC test configuration. + * + */ + @Override + public void deleteOidcTestConfig(DeleteOidcTestConfigReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/oidc_test_configs/{test_slug}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteOidcTestConfigResponse.Builder responseBuilder = DeleteOidcTestConfigResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create a OIDC test configuration. + * + */ + @Override + public void createOidcTestConfig(CreateOidcTestConfigReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/oidc_test_configs", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateOidcTestConfigResponse.Builder responseBuilder = CreateOidcTestConfigResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get password config. + * + */ + @Override + public void passwordConfig(PasswordConfigReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/password_config", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + PasswordConfigResponse.Builder responseBuilder = PasswordConfigResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update password config. + * + */ + @Override + public void updatePasswordConfig(UpdatePasswordConfigReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/password_config", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdatePasswordConfigResponse.Builder responseBuilder = UpdatePasswordConfigResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Force all credentials_email users to reset their login passwords upon their next login. + * + */ + @Override + public void forcePasswordResetAtNextLoginForAllUsers(ForcePasswordResetAtNextLoginForAllUsersReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.put("/password_config/force_password_reset_at_next_login_for_all_users", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ForcePasswordResetAtNextLoginForAllUsersResponse.Builder responseBuilder = ForcePasswordResetAtNextLoginForAllUsersResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get the SAML configuration. + * + * Looker can be optionally configured to authenticate users against a SAML authentication server. + * SAML setup requires coordination with an administrator of that server. + * + * Only Looker administrators can read and update the SAML configuration. + * + * Configuring SAML impacts authentication for all users. This configuration should be done carefully. + * + * Looker maintains a single SAML configuation. It can be read and updated. Updates only succeed if the new state will be valid (in the sense that all required fields are populated); it is up to you to ensure that the configuration is appropriate and correct). + * + * SAML is enabled or disabled for Looker using the **enabled** field. + * + */ + @Override + public void samlConfig(SamlConfigReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/saml_config", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SamlConfigResponse.Builder responseBuilder = SamlConfigResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update the SAML configuration. + * + * Configuring SAML impacts authentication for all users. This configuration should be done carefully. + * + * Only Looker administrators can read and update the SAML configuration. + * + * SAML is enabled or disabled for Looker using the **enabled** field. + * + * It is **highly** recommended that any SAML setting changes be tested using the APIs below before being set globally. + * + */ + @Override + public void updateSamlConfig(UpdateSamlConfigReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/saml_config", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateSamlConfigResponse.Builder responseBuilder = UpdateSamlConfigResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get a SAML test configuration by test_slug. + * + */ + @Override + public void samlTestConfig(SamlTestConfigReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/saml_test_configs/{test_slug}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SamlTestConfigResponse.Builder responseBuilder = SamlTestConfigResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Delete a SAML test configuration. + * + */ + @Override + public void deleteSamlTestConfig(DeleteSamlTestConfigReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/saml_test_configs/{test_slug}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteSamlTestConfigResponse.Builder responseBuilder = DeleteSamlTestConfigResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create a SAML test configuration. + * + */ + @Override + public void createSamlTestConfig(CreateSamlTestConfigReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/saml_test_configs", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateSamlTestConfigResponse.Builder responseBuilder = CreateSamlTestConfigResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Parse the given xml as a SAML IdP metadata document and return the result. + * + */ + @Override + public void parseSamlIdpMetadata(ParseSamlIdpMetadataReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/parse_saml_idp_metadata", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ParseSamlIdpMetadataResponse.Builder responseBuilder = ParseSamlIdpMetadataResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Fetch the given url and parse it as a SAML IdP metadata document and return the result. + * Note that this requires that the url be public or at least at a location where the Looker instance + * can fetch it without requiring any special authentication. + * + */ + @Override + public void fetchAndParseSamlIdpMetadata(FetchAndParseSamlIdpMetadataReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/fetch_and_parse_saml_idp_metadata", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + FetchAndParseSamlIdpMetadataResponse.Builder responseBuilder = FetchAndParseSamlIdpMetadataResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get session config. + * + */ + @Override + public void sessionConfig(SessionConfigReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/session_config", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SessionConfigResponse.Builder responseBuilder = SessionConfigResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update session config. + * + */ + @Override + public void updateSessionConfig(UpdateSessionConfigReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/session_config", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateSessionConfigResponse.Builder responseBuilder = UpdateSessionConfigResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get Support Access Allowlist Users + * + * Returns the users that have been added to the Support Access Allowlist + * + */ + @Override + public void getSupportAccessAllowlistEntries(GetSupportAccessAllowlistEntriesReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/support_access/allowlist", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + GetSupportAccessAllowlistEntriesResponse.Builder responseBuilder = GetSupportAccessAllowlistEntriesResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + GetSupportAccessAllowlistEntriesStreamResponse.Builder responseBuilder2 = GetSupportAccessAllowlistEntriesStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Add Support Access Allowlist Users + * + * Adds a list of emails to the Allowlist, using the provided reason + * + */ + @Override + public void addSupportAccessAllowlistEntries(AddSupportAccessAllowlistEntriesReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/support_access/allowlist", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AddSupportAccessAllowlistEntriesResponse.Builder responseBuilder = AddSupportAccessAllowlistEntriesResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + AddSupportAccessAllowlistEntriesStreamResponse.Builder responseBuilder2 = AddSupportAccessAllowlistEntriesStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Delete Support Access Allowlist User + * + * Deletes the specified Allowlist Entry Id + * + */ + @Override + public void deleteSupportAccessAllowlistEntry(DeleteSupportAccessAllowlistEntryReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/support_access/allowlist/{entry_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteSupportAccessAllowlistEntryResponse.Builder responseBuilder = DeleteSupportAccessAllowlistEntryResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Enable Support Access + * + * Enables Support Access for the provided duration + * + */ + @Override + public void enableSupportAccess(EnableSupportAccessReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.put("/support_access/enable", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + EnableSupportAccessResponse.Builder responseBuilder = EnableSupportAccessResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Disable Support Access + * + * Disables Support Access immediately + * + */ + @Override + public void disableSupportAccess(DisableSupportAccessReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.put("/support_access/disable", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DisableSupportAccessResponse.Builder responseBuilder = DisableSupportAccessResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Support Access Status + * + * Returns the current Support Access Status + * + */ + @Override + public void supportAccessStatus(SupportAccessStatusReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/support_access/status", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SupportAccessStatusResponse.Builder responseBuilder = SupportAccessStatusResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get currently locked-out users. + * + */ + @Override + public void allUserLoginLockouts(AllUserLoginLockoutsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/user_login_lockouts", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllUserLoginLockoutsResponse.Builder responseBuilder = AllUserLoginLockoutsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + AllUserLoginLockoutsStreamResponse.Builder responseBuilder2 = AllUserLoginLockoutsStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Search currently locked-out users. + * + */ + @Override + public void searchUserLoginLockouts(SearchUserLoginLockoutsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/user_login_lockouts/search", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SearchUserLoginLockoutsResponse.Builder responseBuilder = SearchUserLoginLockoutsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + SearchUserLoginLockoutsStreamResponse.Builder responseBuilder2 = SearchUserLoginLockoutsStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Removes login lockout for the associated user. + * + */ + @Override + public void deleteUserLoginLockout(DeleteUserLoginLockoutReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/user_login_lockout/{key}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteUserLoginLockoutResponse.Builder responseBuilder = DeleteUserLoginLockoutResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + //#endregion Auth: Manage User Authentication Configuration + + //#region Board: Manage Boards + + /** + * ### Get information about all boards. + * + */ + @Override + public void allBoards(AllBoardsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/boards", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllBoardsResponse.Builder responseBuilder = AllBoardsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + AllBoardsStreamResponse.Builder responseBuilder2 = AllBoardsStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create a new board. + * + */ + @Override + public void createBoard(CreateBoardReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/boards", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateBoardResponse.Builder responseBuilder = CreateBoardResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Search Boards + * + * If multiple search params are given and `filter_or` is FALSE or not specified, + * search params are combined in a logical AND operation. + * Only rows that match *all* search param criteria will be returned. + * + * If `filter_or` is TRUE, multiple search params are combined in a logical OR operation. + * Results will include rows that match **any** of the search criteria. + * + * String search params use case-insensitive matching. + * String search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions. + * example="dan%" will match "danger" and "Danzig" but not "David" + * example="D_m%" will match "Damage" and "dump" + * + * Integer search params can accept a single value or a comma separated list of values. The multiple + * values will be combined under a logical OR operation - results will match at least one of + * the given values. + * + * Most search params can accept "IS NULL" and "NOT NULL" as special expressions to match + * or exclude (respectively) rows where the column is null. + * + * Boolean search params accept only "true" and "false" as values. + * + * + */ + @Override + public void searchBoards(SearchBoardsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/boards/search", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SearchBoardsResponse.Builder responseBuilder = SearchBoardsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + SearchBoardsStreamResponse.Builder responseBuilder2 = SearchBoardsStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about a board. + * + */ + @Override + public void board(BoardReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/boards/{board_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + BoardResponse.Builder responseBuilder = BoardResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update a board definition. + * + */ + @Override + public void updateBoard(UpdateBoardReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/boards/{board_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateBoardResponse.Builder responseBuilder = UpdateBoardResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Delete a board. + * + */ + @Override + public void deleteBoard(DeleteBoardReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/boards/{board_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteBoardResponse.Builder responseBuilder = DeleteBoardResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about all board items. + * + */ + @Override + public void allBoardItems(AllBoardItemsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/board_items", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllBoardItemsResponse.Builder responseBuilder = AllBoardItemsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + AllBoardItemsStreamResponse.Builder responseBuilder2 = AllBoardItemsStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create a new board item. + * + */ + @Override + public void createBoardItem(CreateBoardItemReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/board_items", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateBoardItemResponse.Builder responseBuilder = CreateBoardItemResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about a board item. + * + */ + @Override + public void boardItem(BoardItemReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/board_items/{board_item_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + BoardItemResponse.Builder responseBuilder = BoardItemResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update a board item definition. + * + */ + @Override + public void updateBoardItem(UpdateBoardItemReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/board_items/{board_item_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateBoardItemResponse.Builder responseBuilder = UpdateBoardItemResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Delete a board item. + * + */ + @Override + public void deleteBoardItem(DeleteBoardItemReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/board_items/{board_item_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteBoardItemResponse.Builder responseBuilder = DeleteBoardItemResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about all board sections. + * + */ + @Override + public void allBoardSections(AllBoardSectionsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/board_sections", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllBoardSectionsResponse.Builder responseBuilder = AllBoardSectionsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + AllBoardSectionsStreamResponse.Builder responseBuilder2 = AllBoardSectionsStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create a new board section. + * + */ + @Override + public void createBoardSection(CreateBoardSectionReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/board_sections", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateBoardSectionResponse.Builder responseBuilder = CreateBoardSectionResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about a board section. + * + */ + @Override + public void boardSection(BoardSectionReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/board_sections/{board_section_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + BoardSectionResponse.Builder responseBuilder = BoardSectionResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update a board section definition. + * + */ + @Override + public void updateBoardSection(UpdateBoardSectionReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/board_sections/{board_section_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateBoardSectionResponse.Builder responseBuilder = UpdateBoardSectionResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Delete a board section. + * + */ + @Override + public void deleteBoardSection(DeleteBoardSectionReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/board_sections/{board_section_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteBoardSectionResponse.Builder responseBuilder = DeleteBoardSectionResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + //#endregion Board: Manage Boards + + //#region ColorCollection: Manage Color Collections + + /** + * ### Get an array of all existing Color Collections + * Get a **single** color collection by id with [ColorCollection](#!/ColorCollection/color_collection) + * + * Get all **standard** color collections with [ColorCollection](#!/ColorCollection/color_collections_standard) + * + * Get all **custom** color collections with [ColorCollection](#!/ColorCollection/color_collections_custom) + * + * **Note**: Only an API user with the Admin role can call this endpoint. Unauthorized requests will return `Not Found` (404) errors. + * + * + */ + @Override + public void allColorCollections(AllColorCollectionsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/color_collections", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllColorCollectionsResponse.Builder responseBuilder = AllColorCollectionsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + AllColorCollectionsStreamResponse.Builder responseBuilder2 = AllColorCollectionsStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create a custom color collection with the specified information + * + * Creates a new custom color collection object, returning the details, including the created id. + * + * **Update** an existing color collection with [Update Color Collection](#!/ColorCollection/update_color_collection) + * + * **Permanently delete** an existing custom color collection with [Delete Color Collection](#!/ColorCollection/delete_color_collection) + * + * **Note**: Only an API user with the Admin role can call this endpoint. Unauthorized requests will return `Not Found` (404) errors. + * + * + */ + @Override + public void createColorCollection(CreateColorCollectionReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/color_collections", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateColorCollectionResponse.Builder responseBuilder = CreateColorCollectionResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get an array of all existing **Custom** Color Collections + * Get a **single** color collection by id with [ColorCollection](#!/ColorCollection/color_collection) + * + * Get all **standard** color collections with [ColorCollection](#!/ColorCollection/color_collections_standard) + * + * **Note**: Only an API user with the Admin role can call this endpoint. Unauthorized requests will return `Not Found` (404) errors. + * + * + */ + @Override + public void colorCollectionsCustom(ColorCollectionsCustomReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/color_collections/custom", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ColorCollectionsCustomResponse.Builder responseBuilder = ColorCollectionsCustomResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + ColorCollectionsCustomStreamResponse.Builder responseBuilder2 = ColorCollectionsCustomStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get an array of all existing **Standard** Color Collections + * Get a **single** color collection by id with [ColorCollection](#!/ColorCollection/color_collection) + * + * Get all **custom** color collections with [ColorCollection](#!/ColorCollection/color_collections_custom) + * + * **Note**: Only an API user with the Admin role can call this endpoint. Unauthorized requests will return `Not Found` (404) errors. + * + * + */ + @Override + public void colorCollectionsStandard(ColorCollectionsStandardReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/color_collections/standard", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ColorCollectionsStandardResponse.Builder responseBuilder = ColorCollectionsStandardResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + ColorCollectionsStandardStreamResponse.Builder responseBuilder2 = ColorCollectionsStandardStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get the default color collection + * + * Use this to retrieve the default Color Collection. + * + * Set the default color collection with [ColorCollection](#!/ColorCollection/set_default_color_collection) + * + */ + @Override + public void defaultColorCollection(DefaultColorCollectionReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/color_collections/default", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DefaultColorCollectionResponse.Builder responseBuilder = DefaultColorCollectionResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Set the global default Color Collection by ID + * + * Returns the new specified default Color Collection object. + * **Note**: Only an API user with the Admin role can call this endpoint. Unauthorized requests will return `Not Found` (404) errors. + * + * + */ + @Override + public void setDefaultColorCollection(SetDefaultColorCollectionReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.put("/color_collections/default", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SetDefaultColorCollectionResponse.Builder responseBuilder = SetDefaultColorCollectionResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get a Color Collection by ID + * + * Use this to retrieve a specific Color Collection. + * Get a **single** color collection by id with [ColorCollection](#!/ColorCollection/color_collection) + * + * Get all **standard** color collections with [ColorCollection](#!/ColorCollection/color_collections_standard) + * + * Get all **custom** color collections with [ColorCollection](#!/ColorCollection/color_collections_custom) + * + * **Note**: Only an API user with the Admin role can call this endpoint. Unauthorized requests will return `Not Found` (404) errors. + * + * + */ + @Override + public void colorCollection(ColorCollectionReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/color_collections/{collection_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ColorCollectionResponse.Builder responseBuilder = ColorCollectionResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update a custom color collection by id. + * **Note**: Only an API user with the Admin role can call this endpoint. Unauthorized requests will return `Not Found` (404) errors. + * + * + */ + @Override + public void updateColorCollection(UpdateColorCollectionReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/color_collections/{collection_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateColorCollectionResponse.Builder responseBuilder = UpdateColorCollectionResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Delete a custom color collection by id + * + * This operation permanently deletes the identified **Custom** color collection. + * + * **Standard** color collections cannot be deleted + * + * Because multiple color collections can have the same label, they must be deleted by ID, not name. + * **Note**: Only an API user with the Admin role can call this endpoint. Unauthorized requests will return `Not Found` (404) errors. + * + * + */ + @Override + public void deleteColorCollection(DeleteColorCollectionReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/color_collections/{collection_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteColorCollectionResponse.Builder responseBuilder = DeleteColorCollectionResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + //#endregion ColorCollection: Manage Color Collections + + //#region Config: Manage General Configuration + + /** + * Get the current Cloud Storage Configuration. + * + */ + @Override + public void cloudStorageConfiguration(CloudStorageConfigurationReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/cloud_storage", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CloudStorageConfigurationResponse.Builder responseBuilder = CloudStorageConfigurationResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * Update the current Cloud Storage Configuration. + * + */ + @Override + public void updateCloudStorageConfiguration(UpdateCloudStorageConfigurationReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/cloud_storage", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateCloudStorageConfigurationResponse.Builder responseBuilder = UpdateCloudStorageConfigurationResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get the current status and content of custom welcome emails + * + */ + @Override + public void customWelcomeEmail(CustomWelcomeEmailReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/custom_welcome_email", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CustomWelcomeEmailResponse.Builder responseBuilder = CustomWelcomeEmailResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * Update custom welcome email setting and values. Optionally send a test email with the new content to the currently logged in user. + * + */ + @Override + public void updateCustomWelcomeEmail(UpdateCustomWelcomeEmailReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/custom_welcome_email", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateCustomWelcomeEmailResponse.Builder responseBuilder = UpdateCustomWelcomeEmailResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * Requests to this endpoint will send a welcome email with the custom content provided in the body to the currently logged in user. + * + */ + @Override + public void updateCustomWelcomeEmailTest(UpdateCustomWelcomeEmailTestReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.put("/custom_welcome_email_test", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateCustomWelcomeEmailTestResponse.Builder responseBuilder = UpdateCustomWelcomeEmailTestResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Retrieve the value for whether or not digest emails is enabled + * + */ + @Override + public void digestEmailsEnabled(DigestEmailsEnabledReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/digest_emails_enabled", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DigestEmailsEnabledResponse.Builder responseBuilder = DigestEmailsEnabledResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update the setting for enabling/disabling digest emails + * + */ + @Override + public void updateDigestEmailsEnabled(UpdateDigestEmailsEnabledReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/digest_emails_enabled", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateDigestEmailsEnabledResponse.Builder responseBuilder = UpdateDigestEmailsEnabledResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Trigger the generation of digest email records and send them to Looker's internal system. This does not send + * any actual emails, it generates records containing content which may be of interest for users who have become inactive. + * Emails will be sent at a later time from Looker's internal system if the Digest Emails feature is enabled in settings. + */ + @Override + public void createDigestEmailSend(CreateDigestEmailSendReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/digest_email_send", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateDigestEmailSendResponse.Builder responseBuilder = CreateDigestEmailSendResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get Egress IP Addresses + * + * Returns the list of public egress IP Addresses for a hosted customer's instance + * + */ + @Override + public void publicEgressIpAddresses(PublicEgressIpAddressesReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/public_egress_ip_addresses", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + PublicEgressIpAddressesResponse.Builder responseBuilder = PublicEgressIpAddressesResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Set the menu item name and content for internal help resources + * + */ + @Override + public void internalHelpResourcesContent(InternalHelpResourcesContentReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/internal_help_resources_content", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + InternalHelpResourcesContentResponse.Builder responseBuilder = InternalHelpResourcesContentResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * Update internal help resources content + * + */ + @Override + public void updateInternalHelpResourcesContent(UpdateInternalHelpResourcesContentReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/internal_help_resources_content", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateInternalHelpResourcesContentResponse.Builder responseBuilder = UpdateInternalHelpResourcesContentResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get and set the options for internal help resources + * + */ + @Override + public void internalHelpResources(InternalHelpResourcesReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/internal_help_resources_enabled", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + InternalHelpResourcesResponse.Builder responseBuilder = InternalHelpResourcesResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * Update internal help resources settings + * + */ + @Override + public void updateInternalHelpResources(UpdateInternalHelpResourcesReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/internal_help_resources", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateInternalHelpResourcesResponse.Builder responseBuilder = UpdateInternalHelpResourcesResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get all legacy features. + * + */ + @Override + public void allLegacyFeatures(AllLegacyFeaturesReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/legacy_features", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllLegacyFeaturesResponse.Builder responseBuilder = AllLegacyFeaturesResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + AllLegacyFeaturesStreamResponse.Builder responseBuilder2 = AllLegacyFeaturesStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about the legacy feature with a specific id. + * + */ + @Override + public void legacyFeature(LegacyFeatureReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/legacy_features/{legacy_feature_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + LegacyFeatureResponse.Builder responseBuilder = LegacyFeatureResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update information about the legacy feature with a specific id. + * + */ + @Override + public void updateLegacyFeature(UpdateLegacyFeatureReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/legacy_features/{legacy_feature_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateLegacyFeatureResponse.Builder responseBuilder = UpdateLegacyFeatureResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get a list of locales that Looker supports. + * + */ + @Override + public void allLocales(AllLocalesReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/locales", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllLocalesResponse.Builder responseBuilder = AllLocalesResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + AllLocalesStreamResponse.Builder responseBuilder2 = AllLocalesStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get all mobile settings. + * + */ + @Override + public void mobileSettings(MobileSettingsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/mobile/settings", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + MobileSettingsResponse.Builder responseBuilder = MobileSettingsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get Looker Settings + * + * Available settings are: + * - extension_framework_enabled + * - marketplace_auto_install_enabled + * - marketplace_enabled + * - privatelabel_configuration + * - custom_welcome_email + * - onboarding_enabled + * + * + */ + @Override + public void getSetting(GetSettingReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/setting", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + GetSettingResponse.Builder responseBuilder = GetSettingResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Configure Looker Settings + * + * Available settings are: + * - extension_framework_enabled + * - marketplace_auto_install_enabled + * - marketplace_enabled + * - privatelabel_configuration + * - custom_welcome_email + * - onboarding_enabled + * + * See the `Setting` type for more information on the specific values that can be configured. + * + */ + @Override + public void setSetting(SetSettingReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/setting", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SetSettingResponse.Builder responseBuilder = SetSettingResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Configure SMTP Settings + * This API allows users to configure the SMTP settings on the Looker instance. + * This API is only supported in the OEM jar. Additionally, only admin users are authorised to call this API. + * + */ + @Override + public void setSmtpSettings(SetSmtpSettingsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/smtp_settings", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SetSmtpSettingsResponse.Builder responseBuilder = SetSmtpSettingsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get current SMTP status. + * + */ + @Override + public void smtpStatus(SmtpStatusReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/smtp_status", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SmtpStatusResponse.Builder responseBuilder = SmtpStatusResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get a list of timezones that Looker supports (e.g. useful for scheduling tasks). + * + */ + @Override + public void allTimezones(AllTimezonesReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/timezones", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllTimezonesResponse.Builder responseBuilder = AllTimezonesResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + AllTimezonesStreamResponse.Builder responseBuilder2 = AllTimezonesStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about all API versions supported by this Looker instance. + * + */ + @Override + public void versions(VersionsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/versions", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + VersionsResponse.Builder responseBuilder = VersionsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get an API specification for this Looker instance. + * + * The specification is returned as a JSON document in Swagger 2.x format + * + */ + @Override + public void apiSpec(ApiSpecReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/api_spec/{api_version}/{specification}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ApiSpecResponse.Builder responseBuilder = ApiSpecResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### This feature is enabled only by special license. + * ### Gets the whitelabel configuration, which includes hiding documentation links, custom favicon uploading, etc. + * + */ + @Override + public void whitelabelConfiguration(WhitelabelConfigurationReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/whitelabel_configuration", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + WhitelabelConfigurationResponse.Builder responseBuilder = WhitelabelConfigurationResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update the whitelabel configuration + * + */ + @Override + public void updateWhitelabelConfiguration(UpdateWhitelabelConfigurationReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.put("/whitelabel_configuration", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateWhitelabelConfigurationResponse.Builder responseBuilder = UpdateWhitelabelConfigurationResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + //#endregion Config: Manage General Configuration + + //#region Connection: Manage Database Connections + + /** + * ### Get information about all connections. + * + */ + @Override + public void allConnections(AllConnectionsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/connections", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllConnectionsResponse.Builder responseBuilder = AllConnectionsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + AllConnectionsStreamResponse.Builder responseBuilder2 = AllConnectionsStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create a connection using the specified configuration. + * + */ + @Override + public void createConnection(CreateConnectionReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/connections", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateConnectionResponse.Builder responseBuilder = CreateConnectionResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about a connection. + * + */ + @Override + public void connection(ConnectionReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/connections/{connection_name}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ConnectionResponse.Builder responseBuilder = ConnectionResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update a connection using the specified configuration. + * + */ + @Override + public void updateConnection(UpdateConnectionReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/connections/{connection_name}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateConnectionResponse.Builder responseBuilder = UpdateConnectionResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Delete a connection. + * + */ + @Override + public void deleteConnection(DeleteConnectionReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/connections/{connection_name}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteConnectionResponse.Builder responseBuilder = DeleteConnectionResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Delete a connection override. + * + */ + @Override + public void deleteConnectionOverride(DeleteConnectionOverrideReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/connections/{connection_name}/connection_override/{override_context}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteConnectionOverrideResponse.Builder responseBuilder = DeleteConnectionOverrideResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Test an existing connection. + * + * Note that a connection's 'dialect' property has a 'connection_tests' property that lists the + * specific types of tests that the connection supports. + * + * This API is rate limited. + * + * Unsupported tests in the request will be ignored. + * + */ + @Override + public void testConnection(TestConnectionReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.put("/connections/{connection_name}/test", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + TestConnectionResponse.Builder responseBuilder = TestConnectionResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + TestConnectionStreamResponse.Builder responseBuilder2 = TestConnectionStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Test a connection configuration. + * + * Note that a connection's 'dialect' property has a 'connection_tests' property that lists the + * specific types of tests that the connection supports. + * + * This API is rate limited. + * + * Unsupported tests in the request will be ignored. + * + */ + @Override + public void testConnectionConfig(TestConnectionConfigReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.put("/connections/test", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + TestConnectionConfigResponse.Builder responseBuilder = TestConnectionConfigResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + TestConnectionConfigStreamResponse.Builder responseBuilder2 = TestConnectionConfigStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about all dialects. + * + */ + @Override + public void allDialectInfos(AllDialectInfosReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/dialect_info", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllDialectInfosResponse.Builder responseBuilder = AllDialectInfosResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + AllDialectInfosStreamResponse.Builder responseBuilder2 = AllDialectInfosStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get all External OAuth Applications. + * + * This is an OAuth Application which Looker uses to access external systems. + * + */ + @Override + public void allExternalOauthApplications(AllExternalOauthApplicationsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/external_oauth_applications", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllExternalOauthApplicationsResponse.Builder responseBuilder = AllExternalOauthApplicationsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + AllExternalOauthApplicationsStreamResponse.Builder responseBuilder2 = AllExternalOauthApplicationsStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create an OAuth Application using the specified configuration. + * + * This is an OAuth Application which Looker uses to access external systems. + * + */ + @Override + public void createExternalOauthApplication(CreateExternalOauthApplicationReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/external_oauth_applications", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateExternalOauthApplicationResponse.Builder responseBuilder = CreateExternalOauthApplicationResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create OAuth User state. + * + */ + @Override + public void createOauthApplicationUserState(CreateOauthApplicationUserStateReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/external_oauth_applications/user_state", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateOauthApplicationUserStateResponse.Builder responseBuilder = CreateOauthApplicationUserStateResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about all SSH Servers. + * + */ + @Override + public void allSshServers(AllSshServersReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/ssh_servers", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllSshServersResponse.Builder responseBuilder = AllSshServersResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + AllSshServersStreamResponse.Builder responseBuilder2 = AllSshServersStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create an SSH Server. + * + */ + @Override + public void createSshServer(CreateSshServerReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/ssh_servers", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateSshServerResponse.Builder responseBuilder = CreateSshServerResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about an SSH Server. + * + */ + @Override + public void sshServer(SshServerReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/ssh_server/{ssh_server_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SshServerResponse.Builder responseBuilder = SshServerResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update an SSH Server. + * + */ + @Override + public void updateSshServer(UpdateSshServerReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/ssh_server/{ssh_server_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateSshServerResponse.Builder responseBuilder = UpdateSshServerResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Delete an SSH Server. + * + */ + @Override + public void deleteSshServer(DeleteSshServerReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/ssh_server/{ssh_server_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteSshServerResponse.Builder responseBuilder = DeleteSshServerResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Test the SSH Server + * + */ + @Override + public void testSshServer(TestSshServerReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/ssh_server/{ssh_server_id}/test", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + TestSshServerResponse.Builder responseBuilder = TestSshServerResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about all SSH Tunnels. + * + */ + @Override + public void allSshTunnels(AllSshTunnelsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/ssh_tunnels", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllSshTunnelsResponse.Builder responseBuilder = AllSshTunnelsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + AllSshTunnelsStreamResponse.Builder responseBuilder2 = AllSshTunnelsStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create an SSH Tunnel + * + */ + @Override + public void createSshTunnel(CreateSshTunnelReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/ssh_tunnels", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateSshTunnelResponse.Builder responseBuilder = CreateSshTunnelResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about an SSH Tunnel. + * + */ + @Override + public void sshTunnel(SshTunnelReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/ssh_tunnel/{ssh_tunnel_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SshTunnelResponse.Builder responseBuilder = SshTunnelResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update an SSH Tunnel + * + */ + @Override + public void updateSshTunnel(UpdateSshTunnelReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/ssh_tunnel/{ssh_tunnel_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateSshTunnelResponse.Builder responseBuilder = UpdateSshTunnelResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Delete an SSH Tunnel + * + */ + @Override + public void deleteSshTunnel(DeleteSshTunnelReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/ssh_tunnel/{ssh_tunnel_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteSshTunnelResponse.Builder responseBuilder = DeleteSshTunnelResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Test the SSH Tunnel + * + */ + @Override + public void testSshTunnel(TestSshTunnelReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/ssh_tunnel/{ssh_tunnel_id}/test", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + TestSshTunnelResponse.Builder responseBuilder = TestSshTunnelResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get the SSH public key + * + * Get the public key created for this instance to identify itself to a remote SSH server. + * + */ + @Override + public void sshPublicKey(SshPublicKeyReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/ssh_public_key", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SshPublicKeyResponse.Builder responseBuilder = SshPublicKeyResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + //#endregion Connection: Manage Database Connections + + //#region Content: Manage Content + + /** + * ### Search Favorite Content + * + * If multiple search params are given and `filter_or` is FALSE or not specified, + * search params are combined in a logical AND operation. + * Only rows that match *all* search param criteria will be returned. + * + * If `filter_or` is TRUE, multiple search params are combined in a logical OR operation. + * Results will include rows that match **any** of the search criteria. + * + * String search params use case-insensitive matching. + * String search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions. + * example="dan%" will match "danger" and "Danzig" but not "David" + * example="D_m%" will match "Damage" and "dump" + * + * Integer search params can accept a single value or a comma separated list of values. The multiple + * values will be combined under a logical OR operation - results will match at least one of + * the given values. + * + * Most search params can accept "IS NULL" and "NOT NULL" as special expressions to match + * or exclude (respectively) rows where the column is null. + * + * Boolean search params accept only "true" and "false" as values. + * + * + */ + @Override + public void searchContentFavorites(SearchContentFavoritesReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/content_favorite/search", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SearchContentFavoritesResponse.Builder responseBuilder = SearchContentFavoritesResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + SearchContentFavoritesStreamResponse.Builder responseBuilder2 = SearchContentFavoritesStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get favorite content by its id + */ + @Override + public void contentFavorite(ContentFavoriteReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/content_favorite/{content_favorite_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ContentFavoriteResponse.Builder responseBuilder = ContentFavoriteResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Delete favorite content + */ + @Override + public void deleteContentFavorite(DeleteContentFavoriteReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/content_favorite/{content_favorite_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteContentFavoriteResponse.Builder responseBuilder = DeleteContentFavoriteResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create favorite content + */ + @Override + public void createContentFavorite(CreateContentFavoriteReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/content_favorite", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateContentFavoriteResponse.Builder responseBuilder = CreateContentFavoriteResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about all content metadata in a space. + * + */ + @Override + public void allContentMetadatas(AllContentMetadatasReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/content_metadata", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllContentMetadatasResponse.Builder responseBuilder = AllContentMetadatasResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + AllContentMetadatasStreamResponse.Builder responseBuilder2 = AllContentMetadatasStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about an individual content metadata record. + * + */ + @Override + public void contentMetadata(ContentMetadataReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/content_metadata/{content_metadata_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ContentMetadataResponse.Builder responseBuilder = ContentMetadataResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Move a piece of content. + * + */ + @Override + public void updateContentMetadata(UpdateContentMetadataReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/content_metadata/{content_metadata_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateContentMetadataResponse.Builder responseBuilder = UpdateContentMetadataResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### All content metadata access records for a content metadata item. + * + */ + @Override + public void allContentMetadataAccesses(AllContentMetadataAccessesReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/content_metadata_access", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllContentMetadataAccessesResponse.Builder responseBuilder = AllContentMetadataAccessesResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + AllContentMetadataAccessesStreamResponse.Builder responseBuilder2 = AllContentMetadataAccessesStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create content metadata access. + * + */ + @Override + public void createContentMetadataAccess(CreateContentMetadataAccessReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/content_metadata_access", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateContentMetadataAccessResponse.Builder responseBuilder = CreateContentMetadataAccessResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update type of access for content metadata. + * + */ + @Override + public void updateContentMetadataAccess(UpdateContentMetadataAccessReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.put("/content_metadata_access/{content_metadata_access_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateContentMetadataAccessResponse.Builder responseBuilder = UpdateContentMetadataAccessResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Remove content metadata access. + * + */ + @Override + public void deleteContentMetadataAccess(DeleteContentMetadataAccessReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/content_metadata_access/{content_metadata_access_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteContentMetadataAccessResponse.Builder responseBuilder = DeleteContentMetadataAccessResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get an image representing the contents of a dashboard or look. + * + * The returned thumbnail is an abstract representation of the contents of a dashbord or look and does not + * reflect the actual data displayed in the respective visualizations. + * + */ + @Override + public void contentThumbnail(ContentThumbnailReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/content_thumbnail/{type}/{resource_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ContentThumbnailResponse.Builder responseBuilder = ContentThumbnailResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Validate All Content + * + * Performs validation of all looks and dashboards + * Returns a list of errors found as well as metadata about the content validation run. + * + */ + @Override + public void contentValidation(ContentValidationReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/content_validation", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ContentValidationResponse.Builder responseBuilder = ContentValidationResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Search Content Views + * + * If multiple search params are given and `filter_or` is FALSE or not specified, + * search params are combined in a logical AND operation. + * Only rows that match *all* search param criteria will be returned. + * + * If `filter_or` is TRUE, multiple search params are combined in a logical OR operation. + * Results will include rows that match **any** of the search criteria. + * + * String search params use case-insensitive matching. + * String search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions. + * example="dan%" will match "danger" and "Danzig" but not "David" + * example="D_m%" will match "Damage" and "dump" + * + * Integer search params can accept a single value or a comma separated list of values. The multiple + * values will be combined under a logical OR operation - results will match at least one of + * the given values. + * + * Most search params can accept "IS NULL" and "NOT NULL" as special expressions to match + * or exclude (respectively) rows where the column is null. + * + * Boolean search params accept only "true" and "false" as values. + * + * + */ + @Override + public void searchContentViews(SearchContentViewsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/content_view/search", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SearchContentViewsResponse.Builder responseBuilder = SearchContentViewsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + SearchContentViewsStreamResponse.Builder responseBuilder2 = SearchContentViewsStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get a vector image representing the contents of a dashboard or look. + * + * # DEPRECATED: Use [content_thumbnail()](#!/Content/content_thumbnail) + * + * The returned thumbnail is an abstract representation of the contents of a dashbord or look and does not + * reflect the actual data displayed in the respective visualizations. + * + */ + @Override + public void vectorThumbnail(VectorThumbnailReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/vector_thumbnail/{type}/{resource_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + VectorThumbnailResponse.Builder responseBuilder = VectorThumbnailResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + //#endregion Content: Manage Content + + //#region Dashboard: Manage Dashboards + + /** + * ### Get information about all active dashboards. + * + * Returns an array of **abbreviated dashboard objects**. Dashboards marked as deleted are excluded from this list. + * + * Get the **full details** of a specific dashboard by id with [dashboard()](#!/Dashboard/dashboard) + * + * Find **deleted dashboards** with [search_dashboards()](#!/Dashboard/search_dashboards) + * + */ + @Override + public void allDashboards(AllDashboardsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/dashboards", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllDashboardsResponse.Builder responseBuilder = AllDashboardsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + AllDashboardsStreamResponse.Builder responseBuilder2 = AllDashboardsStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create a new dashboard + * + * Creates a new dashboard object and returns the details of the newly created dashboard. + * + * `Title` and `space_id` are required fields. + * `Space_id` must contain the id of an existing space. + * A dashboard's `title` must be unique within the space in which it resides. + * + * If you receive a 422 error response when creating a dashboard, be sure to look at the + * response body for information about exactly which fields are missing or contain invalid data. + * + * You can **update** an existing dashboard with [update_dashboard()](#!/Dashboard/update_dashboard) + * + * You can **permanently delete** an existing dashboard with [delete_dashboard()](#!/Dashboard/delete_dashboard) + * + */ + @Override + public void createDashboard(CreateDashboardReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/dashboards", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateDashboardResponse.Builder responseBuilder = CreateDashboardResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Search Dashboards + * + * Returns an **array of dashboard objects** that match the specified search criteria. + * + * If multiple search params are given and `filter_or` is FALSE or not specified, + * search params are combined in a logical AND operation. + * Only rows that match *all* search param criteria will be returned. + * + * If `filter_or` is TRUE, multiple search params are combined in a logical OR operation. + * Results will include rows that match **any** of the search criteria. + * + * String search params use case-insensitive matching. + * String search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions. + * example="dan%" will match "danger" and "Danzig" but not "David" + * example="D_m%" will match "Damage" and "dump" + * + * Integer search params can accept a single value or a comma separated list of values. The multiple + * values will be combined under a logical OR operation - results will match at least one of + * the given values. + * + * Most search params can accept "IS NULL" and "NOT NULL" as special expressions to match + * or exclude (respectively) rows where the column is null. + * + * Boolean search params accept only "true" and "false" as values. + * + * + * The parameters `limit`, and `offset` are recommended for fetching results in page-size chunks. + * + * Get a **single dashboard** by id with [dashboard()](#!/Dashboard/dashboard) + * + */ + @Override + public void searchDashboards(SearchDashboardsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/dashboards/search", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SearchDashboardsResponse.Builder responseBuilder = SearchDashboardsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + SearchDashboardsStreamResponse.Builder responseBuilder2 = SearchDashboardsStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Import a LookML dashboard to a space as a UDD + * Creates a UDD (a dashboard which exists in the Looker database rather than as a LookML file) from the LookML dashboard + * and places it in the space specified. The created UDD will have a lookml_link_id which links to the original LookML dashboard. + * + * To give the imported dashboard specify a (e.g. title: "my title") in the body of your request, otherwise the imported + * dashboard will have the same title as the original LookML dashboard. + * + * For this operation to succeed the user must have permission to see the LookML dashboard in question, and have permission to + * create content in the space the dashboard is being imported to. + * + * **Sync** a linked UDD with [sync_lookml_dashboard()](#!/Dashboard/sync_lookml_dashboard) + * **Unlink** a linked UDD by setting lookml_link_id to null with [update_dashboard()](#!/Dashboard/update_dashboard) + * + */ + @Override + public void importLookmlDashboard(ImportLookmlDashboardReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/dashboards/{lookml_dashboard_id}/import/{space_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ImportLookmlDashboardResponse.Builder responseBuilder = ImportLookmlDashboardResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update all linked dashboards to match the specified LookML dashboard. + * + * Any UDD (a dashboard which exists in the Looker database rather than as a LookML file) which has a `lookml_link_id` + * property value referring to a LookML dashboard's id (model::dashboardname) will be updated so that it matches the current state of the LookML dashboard. + * + * For this operation to succeed the user must have permission to view the LookML dashboard, and only linked dashboards + * that the user has permission to update will be synced. + * + * To **link** or **unlink** a UDD set the `lookml_link_id` property with [update_dashboard()](#!/Dashboard/update_dashboard) + * + */ + @Override + public void syncLookmlDashboard(SyncLookmlDashboardReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/dashboards/{lookml_dashboard_id}/sync", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SyncLookmlDashboardResponse.Builder responseBuilder = SyncLookmlDashboardResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + SyncLookmlDashboardStreamResponse.Builder responseBuilder2 = SyncLookmlDashboardStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about a dashboard + * + * Returns the full details of the identified dashboard object + * + * Get a **summary list** of all active dashboards with [all_dashboards()](#!/Dashboard/all_dashboards) + * + * You can **Search** for dashboards with [search_dashboards()](#!/Dashboard/search_dashboards) + * + */ + @Override + public void dashboard(DashboardReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/dashboards/{dashboard_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DashboardResponse.Builder responseBuilder = DashboardResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update a dashboard + * + * You can use this function to change the string and integer properties of + * a dashboard. Nested objects such as filters, dashboard elements, or dashboard layout components + * cannot be modified by this function - use the update functions for the respective + * nested object types (like [update_dashboard_filter()](#!/3.1/Dashboard/update_dashboard_filter) to change a filter) + * to modify nested objects referenced by a dashboard. + * + * If you receive a 422 error response when updating a dashboard, be sure to look at the + * response body for information about exactly which fields are missing or contain invalid data. + * + */ + @Override + public void updateDashboard(UpdateDashboardReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/dashboards/{dashboard_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateDashboardResponse.Builder responseBuilder = UpdateDashboardResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Delete the dashboard with the specified id + * + * Permanently **deletes** a dashboard. (The dashboard cannot be recovered after this operation.) + * + * "Soft" delete or hide a dashboard by setting its `deleted` status to `True` with [update_dashboard()](#!/Dashboard/update_dashboard). + * + * Note: When a dashboard is deleted in the UI, it is soft deleted. Use this API call to permanently remove it, if desired. + * + */ + @Override + public void deleteDashboard(DeleteDashboardReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/dashboards/{dashboard_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteDashboardResponse.Builder responseBuilder = DeleteDashboardResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get Aggregate Table LookML for Each Query on a Dahboard + * + * Returns a JSON object that contains the dashboard id and Aggregate Table lookml + * + * + */ + @Override + public void dashboardAggregateTableLookml(DashboardAggregateTableLookmlReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/dashboards/aggregate_table_lookml/{dashboard_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DashboardAggregateTableLookmlResponse.Builder responseBuilder = DashboardAggregateTableLookmlResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get lookml of a UDD + * + * Returns a JSON object that contains the dashboard id and the full lookml + * + * + */ + @Override + public void dashboardLookml(DashboardLookmlReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/dashboards/lookml/{dashboard_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DashboardLookmlResponse.Builder responseBuilder = DashboardLookmlResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Move an existing dashboard + * + * Moves a dashboard to a specified folder, and returns the moved dashboard. + * + * `dashboard_id` and `folder_id` are required. + * `dashboard_id` and `folder_id` must already exist, and `folder_id` must be different from the current `folder_id` of the dashboard. + * + */ + @Override + public void moveDashboard(MoveDashboardReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/dashboards/{dashboard_id}/move", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + MoveDashboardResponse.Builder responseBuilder = MoveDashboardResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Creates a new dashboard object based on LookML Dashboard YAML, and returns the details of the newly created dashboard. + * + * This is equivalent to creating a LookML Dashboard and converting to a User-defined dashboard. + * + * LookML must contain valid LookML YAML code. It's recommended to use the LookML format returned + * from [dashboard_lookml()](#!/Dashboard/dashboard_lookml) as the input LookML (newlines replaced with + * ). + * + * Note that the created dashboard is not linked to any LookML Dashboard, + * i.e. [sync_lookml_dashboard()](#!/Dashboard/sync_lookml_dashboard) will not update dashboards created by this method. + * + * + */ + @Override + public void createDashboardFromLookml(CreateDashboardFromLookmlReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/dashboards/from_lookml", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateDashboardFromLookmlResponse.Builder responseBuilder = CreateDashboardFromLookmlResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Copy an existing dashboard + * + * Creates a copy of an existing dashboard, in a specified folder, and returns the copied dashboard. + * + * `dashboard_id` is required, `dashboard_id` and `folder_id` must already exist if specified. + * `folder_id` will default to the existing folder. + * + * If a dashboard with the same title already exists in the target folder, the copy will have '(copy)' + * or '(copy <# of copies>)' appended. + * + */ + @Override + public void copyDashboard(CopyDashboardReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/dashboards/{dashboard_id}/copy", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CopyDashboardResponse.Builder responseBuilder = CopyDashboardResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Search Dashboard Elements + * + * Returns an **array of DashboardElement objects** that match the specified search criteria. + * + * If multiple search params are given and `filter_or` is FALSE or not specified, + * search params are combined in a logical AND operation. + * Only rows that match *all* search param criteria will be returned. + * + * If `filter_or` is TRUE, multiple search params are combined in a logical OR operation. + * Results will include rows that match **any** of the search criteria. + * + * String search params use case-insensitive matching. + * String search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions. + * example="dan%" will match "danger" and "Danzig" but not "David" + * example="D_m%" will match "Damage" and "dump" + * + * Integer search params can accept a single value or a comma separated list of values. The multiple + * values will be combined under a logical OR operation - results will match at least one of + * the given values. + * + * Most search params can accept "IS NULL" and "NOT NULL" as special expressions to match + * or exclude (respectively) rows where the column is null. + * + * Boolean search params accept only "true" and "false" as values. + * + * + */ + @Override + public void searchDashboardElements(SearchDashboardElementsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/dashboard_elements/search", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SearchDashboardElementsResponse.Builder responseBuilder = SearchDashboardElementsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + SearchDashboardElementsStreamResponse.Builder responseBuilder2 = SearchDashboardElementsStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about the dashboard element with a specific id. + */ + @Override + public void dashboardElement(DashboardElementReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/dashboard_elements/{dashboard_element_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DashboardElementResponse.Builder responseBuilder = DashboardElementResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update the dashboard element with a specific id. + */ + @Override + public void updateDashboardElement(UpdateDashboardElementReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/dashboard_elements/{dashboard_element_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateDashboardElementResponse.Builder responseBuilder = UpdateDashboardElementResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Delete a dashboard element with a specific id. + */ + @Override + public void deleteDashboardElement(DeleteDashboardElementReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/dashboard_elements/{dashboard_element_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteDashboardElementResponse.Builder responseBuilder = DeleteDashboardElementResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about all the dashboard elements on a dashboard with a specific id. + */ + @Override + public void dashboardDashboardElements(DashboardDashboardElementsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/dashboards/{dashboard_id}/dashboard_elements", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DashboardDashboardElementsResponse.Builder responseBuilder = DashboardDashboardElementsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + DashboardDashboardElementsStreamResponse.Builder responseBuilder2 = DashboardDashboardElementsStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create a dashboard element on the dashboard with a specific id. + */ + @Override + public void createDashboardElement(CreateDashboardElementReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/dashboard_elements", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateDashboardElementResponse.Builder responseBuilder = CreateDashboardElementResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about the dashboard filters with a specific id. + */ + @Override + public void dashboardFilter(DashboardFilterReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/dashboard_filters/{dashboard_filter_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DashboardFilterResponse.Builder responseBuilder = DashboardFilterResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update the dashboard filter with a specific id. + */ + @Override + public void updateDashboardFilter(UpdateDashboardFilterReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/dashboard_filters/{dashboard_filter_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateDashboardFilterResponse.Builder responseBuilder = UpdateDashboardFilterResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Delete a dashboard filter with a specific id. + */ + @Override + public void deleteDashboardFilter(DeleteDashboardFilterReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/dashboard_filters/{dashboard_filter_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteDashboardFilterResponse.Builder responseBuilder = DeleteDashboardFilterResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about all the dashboard filters on a dashboard with a specific id. + */ + @Override + public void dashboardDashboardFilters(DashboardDashboardFiltersReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/dashboards/{dashboard_id}/dashboard_filters", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DashboardDashboardFiltersResponse.Builder responseBuilder = DashboardDashboardFiltersResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + DashboardDashboardFiltersStreamResponse.Builder responseBuilder2 = DashboardDashboardFiltersStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create a dashboard filter on the dashboard with a specific id. + */ + @Override + public void createDashboardFilter(CreateDashboardFilterReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/dashboard_filters", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateDashboardFilterResponse.Builder responseBuilder = CreateDashboardFilterResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about the dashboard elements with a specific id. + */ + @Override + public void dashboardLayoutComponent(DashboardLayoutComponentReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/dashboard_layout_components/{dashboard_layout_component_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DashboardLayoutComponentResponse.Builder responseBuilder = DashboardLayoutComponentResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update the dashboard element with a specific id. + */ + @Override + public void updateDashboardLayoutComponent(UpdateDashboardLayoutComponentReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/dashboard_layout_components/{dashboard_layout_component_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateDashboardLayoutComponentResponse.Builder responseBuilder = UpdateDashboardLayoutComponentResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about all the dashboard layout components for a dashboard layout with a specific id. + */ + @Override + public void dashboardLayoutDashboardLayoutComponents(DashboardLayoutDashboardLayoutComponentsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/dashboard_layouts/{dashboard_layout_id}/dashboard_layout_components", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DashboardLayoutDashboardLayoutComponentsResponse.Builder responseBuilder = DashboardLayoutDashboardLayoutComponentsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + DashboardLayoutDashboardLayoutComponentsStreamResponse.Builder responseBuilder2 = DashboardLayoutDashboardLayoutComponentsStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about the dashboard layouts with a specific id. + */ + @Override + public void dashboardLayout(DashboardLayoutReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/dashboard_layouts/{dashboard_layout_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DashboardLayoutResponse.Builder responseBuilder = DashboardLayoutResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update the dashboard layout with a specific id. + */ + @Override + public void updateDashboardLayout(UpdateDashboardLayoutReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/dashboard_layouts/{dashboard_layout_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateDashboardLayoutResponse.Builder responseBuilder = UpdateDashboardLayoutResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Delete a dashboard layout with a specific id. + */ + @Override + public void deleteDashboardLayout(DeleteDashboardLayoutReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/dashboard_layouts/{dashboard_layout_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteDashboardLayoutResponse.Builder responseBuilder = DeleteDashboardLayoutResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about all the dashboard elements on a dashboard with a specific id. + */ + @Override + public void dashboardDashboardLayouts(DashboardDashboardLayoutsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/dashboards/{dashboard_id}/dashboard_layouts", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DashboardDashboardLayoutsResponse.Builder responseBuilder = DashboardDashboardLayoutsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + DashboardDashboardLayoutsStreamResponse.Builder responseBuilder2 = DashboardDashboardLayoutsStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create a dashboard layout on the dashboard with a specific id. + */ + @Override + public void createDashboardLayout(CreateDashboardLayoutReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/dashboard_layouts", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateDashboardLayoutResponse.Builder responseBuilder = CreateDashboardLayoutResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + //#endregion Dashboard: Manage Dashboards + + //#region DataAction: Run Data Actions + + /** + * Perform a data action. The data action object can be obtained from query results, and used to perform an arbitrary action. + */ + @Override + public void performDataAction(PerformDataActionReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/data_actions", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + PerformDataActionResponse.Builder responseBuilder = PerformDataActionResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * For some data actions, the remote server may supply a form requesting further user input. This endpoint takes a data action, asks the remote server to generate a form for it, and returns that form to you for presentation to the user. + */ + @Override + public void fetchRemoteDataActionForm(FetchRemoteDataActionFormReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/data_actions/form", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + FetchRemoteDataActionFormResponse.Builder responseBuilder = FetchRemoteDataActionFormResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + //#endregion DataAction: Run Data Actions + + //#region Datagroup: Manage Datagroups + + /** + * ### Get information about all datagroups. + * + */ + @Override + public void allDatagroups(AllDatagroupsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/datagroups", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllDatagroupsResponse.Builder responseBuilder = AllDatagroupsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + AllDatagroupsStreamResponse.Builder responseBuilder2 = AllDatagroupsStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about a datagroup. + * + */ + @Override + public void datagroup(DatagroupReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/datagroups/{datagroup_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DatagroupResponse.Builder responseBuilder = DatagroupResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update a datagroup using the specified params. + * + */ + @Override + public void updateDatagroup(UpdateDatagroupReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/datagroups/{datagroup_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateDatagroupResponse.Builder responseBuilder = UpdateDatagroupResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + //#endregion Datagroup: Manage Datagroups + + //#region DerivedTable: View Derived Table graphs + + /** + * ### Discover information about derived tables + * + */ + @Override + public void graphDerivedTablesForModel(GraphDerivedTablesForModelReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/derived_table/graph/model/{model}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + GraphDerivedTablesForModelResponse.Builder responseBuilder = GraphDerivedTablesForModelResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get the subgraph representing this derived table and its dependencies. + * + */ + @Override + public void graphDerivedTablesForView(GraphDerivedTablesForViewReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/derived_table/graph/view/{view}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + GraphDerivedTablesForViewResponse.Builder responseBuilder = GraphDerivedTablesForViewResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * Enqueue materialization for a PDT with the given model name and view name + */ + @Override + public void startPdtBuild(StartPdtBuildReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/derived_table/{model_name}/{view_name}/start", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + StartPdtBuildResponse.Builder responseBuilder = StartPdtBuildResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * Check status of PDT materialization + */ + @Override + public void checkPdtBuild(CheckPdtBuildReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/derived_table/{materialization_id}/status", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CheckPdtBuildResponse.Builder responseBuilder = CheckPdtBuildResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * Stop a PDT materialization + */ + @Override + public void stopPdtBuild(StopPdtBuildReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/derived_table/{materialization_id}/stop", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + StopPdtBuildResponse.Builder responseBuilder = StopPdtBuildResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + //#endregion DerivedTable: View Derived Table graphs + + //#region Folder: Manage Folders + + /** + * Search for folders by creator id, parent id, name, etc + */ + @Override + public void searchFolders(SearchFoldersReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/folders/search", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SearchFoldersResponse.Builder responseBuilder = SearchFoldersResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + SearchFoldersStreamResponse.Builder responseBuilder2 = SearchFoldersStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about the folder with a specific id. + */ + @Override + public void folder(FolderReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/folders/{folder_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + FolderResponse.Builder responseBuilder = FolderResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update the folder with a specific id. + */ + @Override + public void updateFolder(UpdateFolderReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/folders/{folder_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateFolderResponse.Builder responseBuilder = UpdateFolderResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Delete the folder with a specific id including any children folders. + * **DANGER** this will delete all looks and dashboards in the folder. + * + */ + @Override + public void deleteFolder(DeleteFolderReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/folders/{folder_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteFolderResponse.Builder responseBuilder = DeleteFolderResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about all folders. + * + * In API 3.x, this will not return empty personal folders, unless they belong to the calling user, + * or if they contain soft-deleted content. + * + * In API 4.0+, all personal folders will be returned. + * + * + */ + @Override + public void allFolders(AllFoldersReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/folders", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllFoldersResponse.Builder responseBuilder = AllFoldersResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + AllFoldersStreamResponse.Builder responseBuilder2 = AllFoldersStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create a folder with specified information. + * + * Caller must have permission to edit the parent folder and to create folders, otherwise the request + * returns 404 Not Found. + * + */ + @Override + public void createFolder(CreateFolderReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/folders", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateFolderResponse.Builder responseBuilder = CreateFolderResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get the children of a folder. + */ + @Override + public void folderChildren(FolderChildrenReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/folders/{folder_id}/children", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + FolderChildrenResponse.Builder responseBuilder = FolderChildrenResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + FolderChildrenStreamResponse.Builder responseBuilder2 = FolderChildrenStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Search the children of a folder + */ + @Override + public void folderChildrenSearch(FolderChildrenSearchReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/folders/{folder_id}/children/search", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + FolderChildrenSearchResponse.Builder responseBuilder = FolderChildrenSearchResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + FolderChildrenSearchStreamResponse.Builder responseBuilder2 = FolderChildrenSearchStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get the parent of a folder + */ + @Override + public void folderParent(FolderParentReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/folders/{folder_id}/parent", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + FolderParentResponse.Builder responseBuilder = FolderParentResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get the ancestors of a folder + */ + @Override + public void folderAncestors(FolderAncestorsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/folders/{folder_id}/ancestors", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + FolderAncestorsResponse.Builder responseBuilder = FolderAncestorsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + FolderAncestorsStreamResponse.Builder responseBuilder2 = FolderAncestorsStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get all looks in a folder. + * In API 3.x, this will return all looks in a folder, including looks in the trash. + * In API 4.0+, all looks in a folder will be returned, excluding looks in the trash. + * + */ + @Override + public void folderLooks(FolderLooksReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/folders/{folder_id}/looks", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + FolderLooksResponse.Builder responseBuilder = FolderLooksResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + FolderLooksStreamResponse.Builder responseBuilder2 = FolderLooksStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get the dashboards in a folder + */ + @Override + public void folderDashboards(FolderDashboardsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/folders/{folder_id}/dashboards", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + FolderDashboardsResponse.Builder responseBuilder = FolderDashboardsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + FolderDashboardsStreamResponse.Builder responseBuilder2 = FolderDashboardsStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + //#endregion Folder: Manage Folders + + //#region Group: Manage Groups + + /** + * ### Get information about all groups. + * + */ + @Override + public void allGroups(AllGroupsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/groups", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllGroupsResponse.Builder responseBuilder = AllGroupsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + AllGroupsStreamResponse.Builder responseBuilder2 = AllGroupsStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Creates a new group (admin only). + * + */ + @Override + public void createGroup(CreateGroupReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/groups", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateGroupResponse.Builder responseBuilder = CreateGroupResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Search groups + * + * Returns all group records that match the given search criteria. + * + * If multiple search params are given and `filter_or` is FALSE or not specified, + * search params are combined in a logical AND operation. + * Only rows that match *all* search param criteria will be returned. + * + * If `filter_or` is TRUE, multiple search params are combined in a logical OR operation. + * Results will include rows that match **any** of the search criteria. + * + * String search params use case-insensitive matching. + * String search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions. + * example="dan%" will match "danger" and "Danzig" but not "David" + * example="D_m%" will match "Damage" and "dump" + * + * Integer search params can accept a single value or a comma separated list of values. The multiple + * values will be combined under a logical OR operation - results will match at least one of + * the given values. + * + * Most search params can accept "IS NULL" and "NOT NULL" as special expressions to match + * or exclude (respectively) rows where the column is null. + * + * Boolean search params accept only "true" and "false" as values. + * + * + */ + @Override + public void searchGroups(SearchGroupsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/groups/search", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SearchGroupsResponse.Builder responseBuilder = SearchGroupsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + SearchGroupsStreamResponse.Builder responseBuilder2 = SearchGroupsStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Search groups include roles + * + * Returns all group records that match the given search criteria, and attaches any associated roles. + * + * If multiple search params are given and `filter_or` is FALSE or not specified, + * search params are combined in a logical AND operation. + * Only rows that match *all* search param criteria will be returned. + * + * If `filter_or` is TRUE, multiple search params are combined in a logical OR operation. + * Results will include rows that match **any** of the search criteria. + * + * String search params use case-insensitive matching. + * String search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions. + * example="dan%" will match "danger" and "Danzig" but not "David" + * example="D_m%" will match "Damage" and "dump" + * + * Integer search params can accept a single value or a comma separated list of values. The multiple + * values will be combined under a logical OR operation - results will match at least one of + * the given values. + * + * Most search params can accept "IS NULL" and "NOT NULL" as special expressions to match + * or exclude (respectively) rows where the column is null. + * + * Boolean search params accept only "true" and "false" as values. + * + * + */ + @Override + public void searchGroupsWithRoles(SearchGroupsWithRolesReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/groups/search/with_roles", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SearchGroupsWithRolesResponse.Builder responseBuilder = SearchGroupsWithRolesResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + SearchGroupsWithRolesStreamResponse.Builder responseBuilder2 = SearchGroupsWithRolesStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Search groups include hierarchy + * + * Returns all group records that match the given search criteria, and attaches + * associated role_ids and parent group_ids. + * + * If multiple search params are given and `filter_or` is FALSE or not specified, + * search params are combined in a logical AND operation. + * Only rows that match *all* search param criteria will be returned. + * + * If `filter_or` is TRUE, multiple search params are combined in a logical OR operation. + * Results will include rows that match **any** of the search criteria. + * + * String search params use case-insensitive matching. + * String search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions. + * example="dan%" will match "danger" and "Danzig" but not "David" + * example="D_m%" will match "Damage" and "dump" + * + * Integer search params can accept a single value or a comma separated list of values. The multiple + * values will be combined under a logical OR operation - results will match at least one of + * the given values. + * + * Most search params can accept "IS NULL" and "NOT NULL" as special expressions to match + * or exclude (respectively) rows where the column is null. + * + * Boolean search params accept only "true" and "false" as values. + * + * + */ + @Override + public void searchGroupsWithHierarchy(SearchGroupsWithHierarchyReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/groups/search/with_hierarchy", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SearchGroupsWithHierarchyResponse.Builder responseBuilder = SearchGroupsWithHierarchyResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + SearchGroupsWithHierarchyStreamResponse.Builder responseBuilder2 = SearchGroupsWithHierarchyStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about a group. + * + */ + @Override + public void group(GroupReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/groups/{group_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + GroupResponse.Builder responseBuilder = GroupResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Updates the a group (admin only). + */ + @Override + public void updateGroup(UpdateGroupReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/groups/{group_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateGroupResponse.Builder responseBuilder = UpdateGroupResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Deletes a group (admin only). + * + */ + @Override + public void deleteGroup(DeleteGroupReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/groups/{group_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteGroupResponse.Builder responseBuilder = DeleteGroupResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about all the groups in a group + * + */ + @Override + public void allGroupGroups(AllGroupGroupsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/groups/{group_id}/groups", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllGroupGroupsResponse.Builder responseBuilder = AllGroupGroupsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + AllGroupGroupsStreamResponse.Builder responseBuilder2 = AllGroupGroupsStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Adds a new group to a group. + * + */ + @Override + public void addGroupGroup(AddGroupGroupReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/groups/{group_id}/groups", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AddGroupGroupResponse.Builder responseBuilder = AddGroupGroupResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about all the users directly included in a group. + * + */ + @Override + public void allGroupUsers(AllGroupUsersReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/groups/{group_id}/users", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllGroupUsersResponse.Builder responseBuilder = AllGroupUsersResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + AllGroupUsersStreamResponse.Builder responseBuilder2 = AllGroupUsersStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Adds a new user to a group. + * + */ + @Override + public void addGroupUser(AddGroupUserReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/groups/{group_id}/users", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AddGroupUserResponse.Builder responseBuilder = AddGroupUserResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Removes a user from a group. + * + */ + @Override + public void deleteGroupUser(DeleteGroupUserReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/groups/{group_id}/users/{user_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteGroupUserResponse.Builder responseBuilder = DeleteGroupUserResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Removes a group from a group. + * + */ + @Override + public void deleteGroupFromGroup(DeleteGroupFromGroupReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/groups/{group_id}/groups/{deleting_group_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteGroupFromGroupResponse.Builder responseBuilder = DeleteGroupFromGroupResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Set the value of a user attribute for a group. + * + * For information about how user attribute values are calculated, see [Set User Attribute Group Values](#!/UserAttribute/set_user_attribute_group_values). + * + */ + @Override + public void updateUserAttributeGroupValue(UpdateUserAttributeGroupValueReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/groups/{group_id}/attribute_values/{user_attribute_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateUserAttributeGroupValueResponse.Builder responseBuilder = UpdateUserAttributeGroupValueResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Remove a user attribute value from a group. + * + */ + @Override + public void deleteUserAttributeGroupValue(DeleteUserAttributeGroupValueReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/groups/{group_id}/attribute_values/{user_attribute_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteUserAttributeGroupValueResponse.Builder responseBuilder = DeleteUserAttributeGroupValueResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + //#endregion Group: Manage Groups + + //#region Homepage: Manage Homepage + + /** + * ### Get information about the primary homepage's sections. + * + */ + @Override + public void allPrimaryHomepageSections(AllPrimaryHomepageSectionsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/primary_homepage_sections", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllPrimaryHomepageSectionsResponse.Builder responseBuilder = AllPrimaryHomepageSectionsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + AllPrimaryHomepageSectionsStreamResponse.Builder responseBuilder2 = AllPrimaryHomepageSectionsStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + //#endregion Homepage: Manage Homepage + + //#region Integration: Manage Integrations + + /** + * ### Get information about all Integration Hubs. + * + */ + @Override + public void allIntegrationHubs(AllIntegrationHubsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/integration_hubs", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllIntegrationHubsResponse.Builder responseBuilder = AllIntegrationHubsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + AllIntegrationHubsStreamResponse.Builder responseBuilder2 = AllIntegrationHubsStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create a new Integration Hub. + * + * This API is rate limited to prevent it from being used for SSRF attacks + * + */ + @Override + public void createIntegrationHub(CreateIntegrationHubReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/integration_hubs", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateIntegrationHubResponse.Builder responseBuilder = CreateIntegrationHubResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about a Integration Hub. + * + */ + @Override + public void integrationHub(IntegrationHubReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/integration_hubs/{integration_hub_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + IntegrationHubResponse.Builder responseBuilder = IntegrationHubResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update a Integration Hub definition. + * + * This API is rate limited to prevent it from being used for SSRF attacks + * + */ + @Override + public void updateIntegrationHub(UpdateIntegrationHubReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/integration_hubs/{integration_hub_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateIntegrationHubResponse.Builder responseBuilder = UpdateIntegrationHubResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Delete a Integration Hub. + * + */ + @Override + public void deleteIntegrationHub(DeleteIntegrationHubReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/integration_hubs/{integration_hub_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteIntegrationHubResponse.Builder responseBuilder = DeleteIntegrationHubResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * Accepts the legal agreement for a given integration hub. This only works for integration hubs that have legal_agreement_required set to true and legal_agreement_signed set to false. + */ + @Override + public void acceptIntegrationHubLegalAgreement(AcceptIntegrationHubLegalAgreementReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/integration_hubs/{integration_hub_id}/accept_legal_agreement", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AcceptIntegrationHubLegalAgreementResponse.Builder responseBuilder = AcceptIntegrationHubLegalAgreementResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about all Integrations. + * + */ + @Override + public void allIntegrations(AllIntegrationsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/integrations", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllIntegrationsResponse.Builder responseBuilder = AllIntegrationsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + AllIntegrationsStreamResponse.Builder responseBuilder2 = AllIntegrationsStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about a Integration. + * + */ + @Override + public void integration(IntegrationReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/integrations/{integration_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + IntegrationResponse.Builder responseBuilder = IntegrationResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update parameters on a Integration. + * + */ + @Override + public void updateIntegration(UpdateIntegrationReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/integrations/{integration_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateIntegrationResponse.Builder responseBuilder = UpdateIntegrationResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * Returns the Integration form for presentation to the user. + */ + @Override + public void fetchIntegrationForm(FetchIntegrationFormReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/integrations/{integration_id}/form", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + FetchIntegrationFormResponse.Builder responseBuilder = FetchIntegrationFormResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * Tests the integration to make sure all the settings are working. + */ + @Override + public void testIntegration(TestIntegrationReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/integrations/{integration_id}/test", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + TestIntegrationResponse.Builder responseBuilder = TestIntegrationResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + //#endregion Integration: Manage Integrations + + //#region Look: Run and Manage Looks + + /** + * ### Get information about all active Looks + * + * Returns an array of **abbreviated Look objects** describing all the looks that the caller has access to. Soft-deleted Looks are **not** included. + * + * Get the **full details** of a specific look by id with [look(id)](#!/Look/look) + * + * Find **soft-deleted looks** with [search_looks()](#!/Look/search_looks) + * + */ + @Override + public void allLooks(AllLooksReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/looks", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllLooksResponse.Builder responseBuilder = AllLooksResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + AllLooksStreamResponse.Builder responseBuilder2 = AllLooksStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create a Look + * + * To create a look to display query data, first create the query with [create_query()](#!/Query/create_query) + * then assign the query's id to the `query_id` property in the call to `create_look()`. + * + * To place the look into a particular space, assign the space's id to the `space_id` property + * in the call to `create_look()`. + * + */ + @Override + public void createLook(CreateLookReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/looks", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateLookResponse.Builder responseBuilder = CreateLookResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Search Looks + * + * Returns an **array of Look objects** that match the specified search criteria. + * + * If multiple search params are given and `filter_or` is FALSE or not specified, + * search params are combined in a logical AND operation. + * Only rows that match *all* search param criteria will be returned. + * + * If `filter_or` is TRUE, multiple search params are combined in a logical OR operation. + * Results will include rows that match **any** of the search criteria. + * + * String search params use case-insensitive matching. + * String search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions. + * example="dan%" will match "danger" and "Danzig" but not "David" + * example="D_m%" will match "Damage" and "dump" + * + * Integer search params can accept a single value or a comma separated list of values. The multiple + * values will be combined under a logical OR operation - results will match at least one of + * the given values. + * + * Most search params can accept "IS NULL" and "NOT NULL" as special expressions to match + * or exclude (respectively) rows where the column is null. + * + * Boolean search params accept only "true" and "false" as values. + * + * + * Get a **single look** by id with [look(id)](#!/Look/look) + * + */ + @Override + public void searchLooks(SearchLooksReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/looks/search", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SearchLooksResponse.Builder responseBuilder = SearchLooksResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + SearchLooksStreamResponse.Builder responseBuilder2 = SearchLooksStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get a Look. + * + * Returns detailed information about a Look and its associated Query. + * + * + */ + @Override + public void look(LookReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/looks/{look_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + LookResponse.Builder responseBuilder = LookResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Modify a Look + * + * Use this function to modify parts of a look. Property values given in a call to `update_look` are + * applied to the existing look, so there's no need to include properties whose values are not changing. + * It's best to specify only the properties you want to change and leave everything else out + * of your `update_look` call. **Look properties marked 'read-only' will be ignored.** + * + * When a user deletes a look in the Looker UI, the look data remains in the database but is + * marked with a deleted flag ("soft-deleted"). Soft-deleted looks can be undeleted (by an admin) + * if the delete was in error. + * + * To soft-delete a look via the API, use [update_look()](#!/Look/update_look) to change the look's `deleted` property to `true`. + * You can undelete a look by calling `update_look` to change the look's `deleted` property to `false`. + * + * Soft-deleted looks are excluded from the results of [all_looks()](#!/Look/all_looks) and [search_looks()](#!/Look/search_looks), so they + * essentially disappear from view even though they still reside in the db. + * In API 3.1 and later, you can pass `deleted: true` as a parameter to [search_looks()](#!/3.1/Look/search_looks) to list soft-deleted looks. + * + * NOTE: [delete_look()](#!/Look/delete_look) performs a "hard delete" - the look data is removed from the Looker + * database and destroyed. There is no "undo" for `delete_look()`. + * + */ + @Override + public void updateLook(UpdateLookReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/looks/{look_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateLookResponse.Builder responseBuilder = UpdateLookResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Permanently Delete a Look + * + * This operation **permanently** removes a look from the Looker database. + * + * NOTE: There is no "undo" for this kind of delete. + * + * For information about soft-delete (which can be undone) see [update_look()](#!/Look/update_look). + * + */ + @Override + public void deleteLook(DeleteLookReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/looks/{look_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteLookResponse.Builder responseBuilder = DeleteLookResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Run a Look + * + * Runs a given look's query and returns the results in the requested format. + * + * Supported formats: + * + * | result_format | Description + * | :-----------: | :--- | + * | json | Plain json + * | json_detail | Row data plus metadata describing the fields, pivots, table calcs, and other aspects of the query + * | csv | Comma separated values with a header + * | txt | Tab separated values with a header + * | html | Simple html + * | md | Simple markdown + * | xlsx | MS Excel spreadsheet + * | sql | Returns the generated SQL rather than running the query + * | png | A PNG image of the visualization of the query + * | jpg | A JPG image of the visualization of the query + * + * + * + */ + @Override + public void runLook(RunLookReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/looks/{look_id}/run/{result_format}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + RunLookResponse.Builder responseBuilder = RunLookResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Copy an existing look + * + * Creates a copy of an existing look, in a specified folder, and returns the copied look. + * + * `look_id` and `folder_id` are required. + * + * `look_id` and `folder_id` must already exist, and `folder_id` must be different from the current `folder_id` of the dashboard. + * + */ + @Override + public void copyLook(CopyLookReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/looks/{look_id}/copy", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CopyLookResponse.Builder responseBuilder = CopyLookResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Move an existing look + * + * Moves a look to a specified folder, and returns the moved look. + * + * `look_id` and `folder_id` are required. + * `look_id` and `folder_id` must already exist, and `folder_id` must be different from the current `folder_id` of the dashboard. + * + */ + @Override + public void moveLook(MoveLookReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/looks/{look_id}/move", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + MoveLookResponse.Builder responseBuilder = MoveLookResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + //#endregion Look: Run and Manage Looks + + //#region LookmlModel: Manage LookML Models + + /** + * ### Get information about all lookml models. + * + */ + @Override + public void allLookmlModels(AllLookmlModelsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/lookml_models", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllLookmlModelsResponse.Builder responseBuilder = AllLookmlModelsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + AllLookmlModelsStreamResponse.Builder responseBuilder2 = AllLookmlModelsStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create a lookml model using the specified configuration. + * + */ + @Override + public void createLookmlModel(CreateLookmlModelReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/lookml_models", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateLookmlModelResponse.Builder responseBuilder = CreateLookmlModelResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about a lookml model. + * + */ + @Override + public void lookmlModel(LookmlModelReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/lookml_models/{lookml_model_name}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + LookmlModelResponse.Builder responseBuilder = LookmlModelResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update a lookml model using the specified configuration. + * + */ + @Override + public void updateLookmlModel(UpdateLookmlModelReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/lookml_models/{lookml_model_name}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateLookmlModelResponse.Builder responseBuilder = UpdateLookmlModelResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Delete a lookml model. + * + */ + @Override + public void deleteLookmlModel(DeleteLookmlModelReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/lookml_models/{lookml_model_name}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteLookmlModelResponse.Builder responseBuilder = DeleteLookmlModelResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about a lookml model explore. + * + */ + @Override + public void lookmlModelExplore(LookmlModelExploreReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/lookml_models/{lookml_model_name}/explores/{explore_name}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + LookmlModelExploreResponse.Builder responseBuilder = LookmlModelExploreResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + //#endregion LookmlModel: Manage LookML Models + + //#region Metadata: Connection Metadata Features + + /** + * ### Field name suggestions for a model and view + * + * `filters` is a string hash of values, with the key as the field name and the string value as the filter expression: + * + * ```ruby + * {'users.age': '>=60'} + * ``` + * + * or + * + * ```ruby + * {'users.age': '<30'} + * ``` + * + * or + * + * ```ruby + * {'users.age': '=50'} + * ``` + * + */ + @Override + public void modelFieldnameSuggestions(ModelFieldnameSuggestionsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/models/{model_name}/views/{view_name}/fields/{field_name}/suggestions", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ModelFieldnameSuggestionsResponse.Builder responseBuilder = ModelFieldnameSuggestionsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get a single model + * + * + */ + @Override + public void getModel(GetModelReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/models/{model_name}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + GetModelResponse.Builder responseBuilder = GetModelResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### List databases available to this connection + * + * Certain dialects can support multiple databases per single connection. + * If this connection supports multiple databases, the database names will be returned in an array. + * + * Connections using dialects that do not support multiple databases will return an empty array. + * + * **Note**: [Connection Features](#!/Metadata/connection_features) can be used to determine if a connection supports + * multiple databases. + * + */ + @Override + public void connectionDatabases(ConnectionDatabasesReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/connections/{connection_name}/databases", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ConnectionDatabasesResponse.Builder responseBuilder = ConnectionDatabasesResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + ConnectionDatabasesStreamResponse.Builder responseBuilder2 = ConnectionDatabasesStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Retrieve metadata features for this connection + * + * Returns a list of feature names with `true` (available) or `false` (not available) + * + * + */ + @Override + public void connectionFeatures(ConnectionFeaturesReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/connections/{connection_name}/features", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ConnectionFeaturesResponse.Builder responseBuilder = ConnectionFeaturesResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get the list of schemas and tables for a connection + * + * + */ + @Override + public void connectionSchemas(ConnectionSchemasReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/connections/{connection_name}/schemas", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ConnectionSchemasResponse.Builder responseBuilder = ConnectionSchemasResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + ConnectionSchemasStreamResponse.Builder responseBuilder2 = ConnectionSchemasStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get the list of tables for a schema + * + * For dialects that support multiple databases, optionally identify which to use. If not provided, the default + * database for the connection will be used. + * + * For dialects that do **not** support multiple databases, **do not use** the database parameter + * + */ + @Override + public void connectionTables(ConnectionTablesReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/connections/{connection_name}/tables", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ConnectionTablesResponse.Builder responseBuilder = ConnectionTablesResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + ConnectionTablesStreamResponse.Builder responseBuilder2 = ConnectionTablesStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get the columns (and therefore also the tables) in a specific schema + * + * + */ + @Override + public void connectionColumns(ConnectionColumnsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/connections/{connection_name}/columns", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ConnectionColumnsResponse.Builder responseBuilder = ConnectionColumnsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + ConnectionColumnsStreamResponse.Builder responseBuilder2 = ConnectionColumnsStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Search a connection for columns matching the specified name + * + * **Note**: `column_name` must be a valid column name. It is not a search pattern. + * + */ + @Override + public void connectionSearchColumns(ConnectionSearchColumnsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/connections/{connection_name}/search_columns", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ConnectionSearchColumnsResponse.Builder responseBuilder = ConnectionSearchColumnsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + ConnectionSearchColumnsStreamResponse.Builder responseBuilder2 = ConnectionSearchColumnsStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Connection cost estimating + * + * Assign a `sql` statement to the body of the request. e.g., for Ruby, `{sql: 'select * from users'}` + * + * **Note**: If the connection's dialect has no support for cost estimates, an error will be returned + * + */ + @Override + public void connectionCostEstimate(ConnectionCostEstimateReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/connections/{connection_name}/cost_estimate", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ConnectionCostEstimateResponse.Builder responseBuilder = ConnectionCostEstimateResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + //#endregion Metadata: Connection Metadata Features + + //#region Project: Manage Projects + + /** + * ### Generate Lockfile for All LookML Dependencies + * + * Git must have been configured, must be in dev mode and deploy permission required + * + * Install_all is a two step process + * 1. For each remote_dependency in a project the dependency manager will resolve any ambiguous ref. + * 2. The project will then write out a lockfile including each remote_dependency with its resolved ref. + * + * + */ + @Override + public void lockAll(LockAllReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/projects/{project_id}/manifest/lock_all", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + LockAllResponse.Builder responseBuilder = LockAllResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get All Git Branches + * + * Returns a list of git branches in the project repository + * + */ + @Override + public void allGitBranches(AllGitBranchesReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/projects/{project_id}/git_branches", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllGitBranchesResponse.Builder responseBuilder = AllGitBranchesResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + AllGitBranchesStreamResponse.Builder responseBuilder2 = AllGitBranchesStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get the Current Git Branch + * + * Returns the git branch currently checked out in the given project repository + * + */ + @Override + public void gitBranch(GitBranchReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/projects/{project_id}/git_branch", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + GitBranchResponse.Builder responseBuilder = GitBranchResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Checkout and/or reset --hard an existing Git Branch + * + * Only allowed in development mode + * - Call `update_session` to select the 'dev' workspace. + * + * Checkout an existing branch if name field is different from the name of the currently checked out branch. + * + * Optionally specify a branch name, tag name or commit SHA to which the branch should be reset. + * **DANGER** hard reset will be force pushed to the remote. Unsaved changes and commits may be permanently lost. + * + * + */ + @Override + public void updateGitBranch(UpdateGitBranchReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.put("/projects/{project_id}/git_branch", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateGitBranchResponse.Builder responseBuilder = UpdateGitBranchResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create and Checkout a Git Branch + * + * Creates and checks out a new branch in the given project repository + * Only allowed in development mode + * - Call `update_session` to select the 'dev' workspace. + * + * Optionally specify a branch name, tag name or commit SHA as the start point in the ref field. + * If no ref is specified, HEAD of the current branch will be used as the start point for the new branch. + * + * + */ + @Override + public void createGitBranch(CreateGitBranchReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/projects/{project_id}/git_branch", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateGitBranchResponse.Builder responseBuilder = CreateGitBranchResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get the specified Git Branch + * + * Returns the git branch specified in branch_name path param if it exists in the given project repository + * + */ + @Override + public void findGitBranch(FindGitBranchReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/projects/{project_id}/git_branch/{branch_name}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + FindGitBranchResponse.Builder responseBuilder = FindGitBranchResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Delete the specified Git Branch + * + * Delete git branch specified in branch_name path param from local and remote of specified project repository + * + */ + @Override + public void deleteGitBranch(DeleteGitBranchReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/projects/{project_id}/git_branch/{branch_name}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteGitBranchResponse.Builder responseBuilder = DeleteGitBranchResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Deploy a Remote Branch or Ref to Production + * + * Git must have been configured and deploy permission required. + * + * Deploy is a one/two step process + * 1. If this is the first deploy of this project, create the production project with git repository. + * 2. Pull the branch or ref into the production project. + * + * Can only specify either a branch or a ref. + * + * + */ + @Override + public void deployRefToProduction(DeployRefToProductionReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/projects/{project_id}/deploy_ref_to_production", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeployRefToProductionResponse.Builder responseBuilder = DeployRefToProductionResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Deploy LookML from this Development Mode Project to Production + * + * Git must have been configured, must be in dev mode and deploy permission required + * + * Deploy is a two / three step process: + * + * 1. Push commits in current branch of dev mode project to the production branch (origin/master). + * Note a. This step is skipped in read-only projects. + * Note b. If this step is unsuccessful for any reason (e.g. rejected non-fastforward because production branch has + * commits not in current branch), subsequent steps will be skipped. + * 2. If this is the first deploy of this project, create the production project with git repository. + * 3. Pull the production branch into the production project. + * + * + */ + @Override + public void deployToProduction(DeployToProductionReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/projects/{project_id}/deploy_to_production", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeployToProductionResponse.Builder responseBuilder = DeployToProductionResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Reset a project to the revision of the project that is in production. + * + * **DANGER** this will delete any changes that have not been pushed to a remote repository. + * + */ + @Override + public void resetProjectToProduction(ResetProjectToProductionReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/projects/{project_id}/reset_to_production", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ResetProjectToProductionResponse.Builder responseBuilder = ResetProjectToProductionResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Reset a project development branch to the revision of the project that is on the remote. + * + * **DANGER** this will delete any changes that have not been pushed to a remote repository. + * + */ + @Override + public void resetProjectToRemote(ResetProjectToRemoteReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/projects/{project_id}/reset_to_remote", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ResetProjectToRemoteResponse.Builder responseBuilder = ResetProjectToRemoteResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get All Projects + * + * Returns all projects visible to the current user + * + */ + @Override + public void allProjects(AllProjectsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/projects", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllProjectsResponse.Builder responseBuilder = AllProjectsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + AllProjectsStreamResponse.Builder responseBuilder2 = AllProjectsStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create A Project + * + * dev mode required. + * - Call `update_session` to select the 'dev' workspace. + * + * `name` is required. + * `git_remote_url` is not allowed. To configure Git for the newly created project, follow the instructions in `update_project`. + * + * + */ + @Override + public void createProject(CreateProjectReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/projects", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateProjectResponse.Builder responseBuilder = CreateProjectResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get A Project + * + * Returns the project with the given project id + * + */ + @Override + public void project(ProjectReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/projects/{project_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ProjectResponse.Builder responseBuilder = ProjectResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update Project Configuration + * + * Apply changes to a project's configuration. + * + * + * #### Configuring Git for a Project + * + * To set up a Looker project with a remote git repository, follow these steps: + * + * 1. Call `update_session` to select the 'dev' workspace. + * 1. Call `create_git_deploy_key` to create a new deploy key for the project + * 1. Copy the deploy key text into the remote git repository's ssh key configuration + * 1. Call `update_project` to set project's `git_remote_url` ()and `git_service_name`, if necessary). + * + * When you modify a project's `git_remote_url`, Looker connects to the remote repository to fetch + * metadata. The remote git repository MUST be configured with the Looker-generated deploy + * key for this project prior to setting the project's `git_remote_url`. + * + * To set up a Looker project with a git repository residing on the Looker server (a 'bare' git repo): + * + * 1. Call `update_session` to select the 'dev' workspace. + * 1. Call `update_project` setting `git_remote_url` to null and `git_service_name` to "bare". + * + * + */ + @Override + public void updateProject(UpdateProjectReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/projects/{project_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateProjectResponse.Builder responseBuilder = UpdateProjectResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get A Projects Manifest object + * + * Returns the project with the given project id + * + */ + @Override + public void manifest(ManifestReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/projects/{project_id}/manifest", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ManifestResponse.Builder responseBuilder = ManifestResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Git Deploy Key + * + * Returns the ssh public key previously created for a project's git repository. + * + */ + @Override + public void gitDeployKey(GitDeployKeyReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/projects/{project_id}/git/deploy_key", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + GitDeployKeyResponse.Builder responseBuilder = GitDeployKeyResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create Git Deploy Key + * + * Create a public/private key pair for authenticating ssh git requests from Looker to a remote git repository + * for a particular Looker project. + * + * Returns the public key of the generated ssh key pair. + * + * Copy this public key to your remote git repository's ssh keys configuration so that the remote git service can + * validate and accept git requests from the Looker server. + * + */ + @Override + public void createGitDeployKey(CreateGitDeployKeyReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/projects/{project_id}/git/deploy_key", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateGitDeployKeyResponse.Builder responseBuilder = CreateGitDeployKeyResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get Cached Project Validation Results + * + * Returns the cached results of a previous project validation calculation, if any. + * Returns http status 204 No Content if no validation results exist. + * + * Validating the content of all the files in a project can be computationally intensive + * for large projects. Use this API to simply fetch the results of the most recent + * project validation rather than revalidating the entire project from scratch. + * + * A value of `"stale": true` in the response indicates that the project has changed since + * the cached validation results were computed. The cached validation results may no longer + * reflect the current state of the project. + * + */ + @Override + public void projectValidationResults(ProjectValidationResultsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/projects/{project_id}/validate", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ProjectValidationResultsResponse.Builder responseBuilder = ProjectValidationResultsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Validate Project + * + * Performs lint validation of all lookml files in the project. + * Returns a list of errors found, if any. + * + * Validating the content of all the files in a project can be computationally intensive + * for large projects. For best performance, call `validate_project(project_id)` only + * when you really want to recompute project validation. To quickly display the results of + * the most recent project validation (without recomputing), use `project_validation_results(project_id)` + * + */ + @Override + public void validateProject(ValidateProjectReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/projects/{project_id}/validate", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ValidateProjectResponse.Builder responseBuilder = ValidateProjectResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get Project Workspace + * + * Returns information about the state of the project files in the currently selected workspace + * + */ + @Override + public void projectWorkspace(ProjectWorkspaceReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/projects/{project_id}/current_workspace", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ProjectWorkspaceResponse.Builder responseBuilder = ProjectWorkspaceResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get All Project Files + * + * Returns a list of the files in the project + * + */ + @Override + public void allProjectFiles(AllProjectFilesReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/projects/{project_id}/files", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllProjectFilesResponse.Builder responseBuilder = AllProjectFilesResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + AllProjectFilesStreamResponse.Builder responseBuilder2 = AllProjectFilesStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get Project File Info + * + * Returns information about a file in the project + * + */ + @Override + public void projectFile(ProjectFileReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/projects/{project_id}/files/file", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ProjectFileResponse.Builder responseBuilder = ProjectFileResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get All Git Connection Tests + * + * dev mode required. + * - Call `update_session` to select the 'dev' workspace. + * + * Returns a list of tests which can be run against a project's (or the dependency project for the provided remote_url) git connection. Call [Run Git Connection Test](#!/Project/run_git_connection_test) to execute each test in sequence. + * + * Tests are ordered by increasing specificity. Tests should be run in the order returned because later tests require functionality tested by tests earlier in the test list. + * + * For example, a late-stage test for write access is meaningless if connecting to the git server (an early test) is failing. + * + */ + @Override + public void allGitConnectionTests(AllGitConnectionTestsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/projects/{project_id}/git_connection_tests", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllGitConnectionTestsResponse.Builder responseBuilder = AllGitConnectionTestsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + AllGitConnectionTestsStreamResponse.Builder responseBuilder2 = AllGitConnectionTestsStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Run a git connection test + * + * Run the named test on the git service used by this project (or the dependency project for the provided remote_url) and return the result. This + * is intended to help debug git connections when things do not work properly, to give + * more helpful information about why a git url is not working with Looker. + * + * Tests should be run in the order they are returned by [Get All Git Connection Tests](#!/Project/all_git_connection_tests). + * + */ + @Override + public void runGitConnectionTest(RunGitConnectionTestReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/projects/{project_id}/git_connection_tests/{test_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + RunGitConnectionTestResponse.Builder responseBuilder = RunGitConnectionTestResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get All LookML Tests + * + * Returns a list of tests which can be run to validate a project's LookML code and/or the underlying data, + * optionally filtered by the file id. + * Call [Run LookML Test](#!/Project/run_lookml_test) to execute tests. + * + */ + @Override + public void allLookmlTests(AllLookmlTestsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/projects/{project_id}/lookml_tests", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllLookmlTestsResponse.Builder responseBuilder = AllLookmlTestsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + AllLookmlTestsStreamResponse.Builder responseBuilder2 = AllLookmlTestsStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Run LookML Tests + * + * Runs all tests in the project, optionally filtered by file, test, and/or model. + * + */ + @Override + public void runLookmlTest(RunLookmlTestReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/projects/{project_id}/lookml_tests/run", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + RunLookmlTestResponse.Builder responseBuilder = RunLookmlTestResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + RunLookmlTestStreamResponse.Builder responseBuilder2 = RunLookmlTestStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Creates a tag for the most recent commit, or a specific ref is a SHA is provided + * + * This is an internal-only, undocumented route. + * + */ + @Override + public void tagRef(TagRefReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/projects/{project_id}/tag", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + TagRefResponse.Builder responseBuilder = TagRefResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Configure Repository Credential for a remote dependency + * + * Admin required. + * + * `root_project_id` is required. + * `credential_id` is required. + * + * + */ + @Override + public void updateRepositoryCredential(UpdateRepositoryCredentialReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.put("/projects/{root_project_id}/credential/{credential_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateRepositoryCredentialResponse.Builder responseBuilder = UpdateRepositoryCredentialResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Repository Credential for a remote dependency + * + * Admin required. + * + * `root_project_id` is required. + * `credential_id` is required. + * + */ + @Override + public void deleteRepositoryCredential(DeleteRepositoryCredentialReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/projects/{root_project_id}/credential/{credential_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteRepositoryCredentialResponse.Builder responseBuilder = DeleteRepositoryCredentialResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get all Repository Credentials for a project + * + * `root_project_id` is required. + * + */ + @Override + public void getAllRepositoryCredentials(GetAllRepositoryCredentialsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/projects/{root_project_id}/credentials", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + GetAllRepositoryCredentialsResponse.Builder responseBuilder = GetAllRepositoryCredentialsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + GetAllRepositoryCredentialsStreamResponse.Builder responseBuilder2 = GetAllRepositoryCredentialsStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + //#endregion Project: Manage Projects + + //#region Query: Run and Manage Queries + + /** + * ### Create an async query task + * + * Creates a query task (job) to run a previously created query asynchronously. Returns a Query Task ID. + * + * Use [query_task(query_task_id)](#!/Query/query_task) to check the execution status of the query task. + * After the query task status reaches "Complete", use [query_task_results(query_task_id)](#!/Query/query_task_results) to fetch the results of the query. + * + */ + @Override + public void createQueryTask(CreateQueryTaskReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/query_tasks", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateQueryTaskResponse.Builder responseBuilder = CreateQueryTaskResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Fetch results of multiple async queries + * + * Returns the results of multiple async queries in one request. + * + * For Query Tasks that are not completed, the response will include the execution status of the Query Task but will not include query results. + * Query Tasks whose results have expired will have a status of 'expired'. + * If the user making the API request does not have sufficient privileges to view a Query Task result, the result will have a status of 'missing' + * + */ + @Override + public void queryTaskMultiResults(QueryTaskMultiResultsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/query_tasks/multi_results", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + QueryTaskMultiResultsResponse.Builder responseBuilder = QueryTaskMultiResultsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get Query Task details + * + * Use this function to check the status of an async query task. After the status + * reaches "Complete", you can call [query_task_results(query_task_id)](#!/Query/query_task_results) to + * retrieve the results of the query. + * + * Use [create_query_task()](#!/Query/create_query_task) to create an async query task. + * + */ + @Override + public void queryTask(QueryTaskReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/query_tasks/{query_task_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + QueryTaskResponse.Builder responseBuilder = QueryTaskResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get Async Query Results + * + * Returns the results of an async query task if the query has completed. + * + * If the query task is still running or waiting to run, this function returns 204 No Content. + * + * If the query task ID is invalid or the cached results of the query task have expired, this function returns 404 Not Found. + * + * Use [query_task(query_task_id)](#!/Query/query_task) to check the execution status of the query task + * Call query_task_results only after the query task status reaches "Complete". + * + * You can also use [query_task_multi_results()](#!/Query/query_task_multi_results) retrieve the + * results of multiple async query tasks at the same time. + * + * #### SQL Error Handling: + * If the query fails due to a SQL db error, how this is communicated depends on the result_format you requested in `create_query_task()`. + * + * For `json_detail` result_format: `query_task_results()` will respond with HTTP status '200 OK' and db SQL error info + * will be in the `errors` property of the response object. The 'data' property will be empty. + * + * For all other result formats: `query_task_results()` will respond with HTTP status `400 Bad Request` and some db SQL error info + * will be in the message of the 400 error response, but not as detailed as expressed in `json_detail.errors`. + * These data formats can only carry row data, and error info is not row data. + * + */ + @Override + public void queryTaskResults(QueryTaskResultsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/query_tasks/{query_task_id}/results", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + QueryTaskResultsResponse.Builder responseBuilder = QueryTaskResultsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get a previously created query by id. + * + * A Looker query object includes the various parameters that define a database query that has been run or + * could be run in the future. These parameters include: model, view, fields, filters, pivots, etc. + * Query *results* are not part of the query object. + * + * Query objects are unique and immutable. Query objects are created automatically in Looker as users explore data. + * Looker does not delete them; they become part of the query history. When asked to create a query for + * any given set of parameters, Looker will first try to find an existing query object with matching + * parameters and will only create a new object when an appropriate object can not be found. + * + * This 'get' method is used to get the details about a query for a given id. See the other methods here + * to 'create' and 'run' queries. + * + * Note that some fields like 'filter_config' and 'vis_config' etc are specific to how the Looker UI + * builds queries and visualizations and are not generally useful for API use. They are not required when + * creating new queries and can usually just be ignored. + * + * + */ + @Override + public void query(QueryReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/queries/{query_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + QueryResponse.Builder responseBuilder = QueryResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get the query for a given query slug. + * + * This returns the query for the 'slug' in a query share URL. + * + * The 'slug' is a randomly chosen short string that is used as an alternative to the query's id value + * for use in URLs etc. This method exists as a convenience to help you use the API to 'find' queries that + * have been created using the Looker UI. + * + * You can use the Looker explore page to build a query and then choose the 'Share' option to + * show the share url for the query. Share urls generally look something like 'https://looker.yourcompany/x/vwGSbfc'. + * The trailing 'vwGSbfc' is the share slug. You can pass that string to this api method to get details about the query. + * Those details include the 'id' that you can use to run the query. Or, you can copy the query body + * (perhaps with your own modification) and use that as the basis to make/run new queries. + * + * This will also work with slugs from Looker explore urls like + * 'https://looker.yourcompany/explore/ecommerce/orders?qid=aogBgL6o3cKK1jN3RoZl5s'. In this case + * 'aogBgL6o3cKK1jN3RoZl5s' is the slug. + * + */ + @Override + public void queryForSlug(QueryForSlugReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/queries/slug/{slug}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + QueryForSlugResponse.Builder responseBuilder = QueryForSlugResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create a query. + * + * This allows you to create a new query that you can later run. Looker queries are immutable once created + * and are not deleted. If you create a query that is exactly like an existing query then the existing query + * will be returned and no new query will be created. Whether a new query is created or not, you can use + * the 'id' in the returned query with the 'run' method. + * + * The query parameters are passed as json in the body of the request. + * + * + */ + @Override + public void createQuery(CreateQueryReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/queries", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateQueryResponse.Builder responseBuilder = CreateQueryResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Run a saved query. + * + * This runs a previously saved query. You can use this on a query that was generated in the Looker UI + * or one that you have explicitly created using the API. You can also use a query 'id' from a saved 'Look'. + * + * The 'result_format' parameter specifies the desired structure and format of the response. + * + * Supported formats: + * + * | result_format | Description + * | :-----------: | :--- | + * | json | Plain json + * | json_detail | Row data plus metadata describing the fields, pivots, table calcs, and other aspects of the query + * | csv | Comma separated values with a header + * | txt | Tab separated values with a header + * | html | Simple html + * | md | Simple markdown + * | xlsx | MS Excel spreadsheet + * | sql | Returns the generated SQL rather than running the query + * | png | A PNG image of the visualization of the query + * | jpg | A JPG image of the visualization of the query + * + * + * + */ + @Override + public void runQuery(RunQueryReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/queries/{query_id}/run/{result_format}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + RunQueryResponse.Builder responseBuilder = RunQueryResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Run the query that is specified inline in the posted body. + * + * This allows running a query as defined in json in the posted body. This combines + * the two actions of posting & running a query into one step. + * + * Here is an example body in json: + * ``` + * { + * "model":"thelook", + * "view":"inventory_items", + * "fields":["category.name","inventory_items.days_in_inventory_tier","products.count"], + * "filters":{"category.name":"socks"}, + * "sorts":["products.count desc 0"], + * "limit":"500", + * "query_timezone":"America/Los_Angeles" + * } + * ``` + * + * When using the Ruby SDK this would be passed as a Ruby hash like: + * ``` + * { + * :model=>"thelook", + * :view=>"inventory_items", + * :fields=> + * ["category.name", + * "inventory_items.days_in_inventory_tier", + * "products.count"], + * :filters=>{:"category.name"=>"socks"}, + * :sorts=>["products.count desc 0"], + * :limit=>"500", + * :query_timezone=>"America/Los_Angeles", + * } + * ``` + * + * This will return the result of running the query in the format specified by the 'result_format' parameter. + * + * Supported formats: + * + * | result_format | Description + * | :-----------: | :--- | + * | json | Plain json + * | json_detail | Row data plus metadata describing the fields, pivots, table calcs, and other aspects of the query + * | csv | Comma separated values with a header + * | txt | Tab separated values with a header + * | html | Simple html + * | md | Simple markdown + * | xlsx | MS Excel spreadsheet + * | sql | Returns the generated SQL rather than running the query + * | png | A PNG image of the visualization of the query + * | jpg | A JPG image of the visualization of the query + * + * + * + */ + @Override + public void runInlineQuery(RunInlineQueryReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/queries/run/{result_format}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + RunInlineQueryResponse.Builder responseBuilder = RunInlineQueryResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Run an URL encoded query. + * + * This requires the caller to encode the specifiers for the query into the URL query part using + * Looker-specific syntax as explained below. + * + * Generally, you would want to use one of the methods that takes the parameters as json in the POST body + * for creating and/or running queries. This method exists for cases where one really needs to encode the + * parameters into the URL of a single 'GET' request. This matches the way that the Looker UI formats + * 'explore' URLs etc. + * + * The parameters here are very similar to the json body formatting except that the filter syntax is + * tricky. Unfortunately, this format makes this method not currently callable via the 'Try it out!' button + * in this documentation page. But, this is callable when creating URLs manually or when using the Looker SDK. + * + * Here is an example inline query URL: + * + * ``` + * https://looker.mycompany.com:19999/api/3.0/queries/models/thelook/views/inventory_items/run/json?fields=category.name,inventory_items.days_in_inventory_tier,products.count&f[category.name]=socks&sorts=products.count+desc+0&limit=500&query_timezone=America/Los_Angeles + * ``` + * + * When invoking this endpoint with the Ruby SDK, pass the query parameter parts as a hash. The hash to match the above would look like: + * + * ```ruby + * query_params = + * { + * fields: "category.name,inventory_items.days_in_inventory_tier,products.count", + * :"f[category.name]" => "socks", + * sorts: "products.count desc 0", + * limit: "500", + * query_timezone: "America/Los_Angeles" + * } + * response = ruby_sdk.run_url_encoded_query('thelook','inventory_items','json', query_params) + * + * ``` + * + * Again, it is generally easier to use the variant of this method that passes the full query in the POST body. + * This method is available for cases where other alternatives won't fit the need. + * + * Supported formats: + * + * | result_format | Description + * | :-----------: | :--- | + * | json | Plain json + * | json_detail | Row data plus metadata describing the fields, pivots, table calcs, and other aspects of the query + * | csv | Comma separated values with a header + * | txt | Tab separated values with a header + * | html | Simple html + * | md | Simple markdown + * | xlsx | MS Excel spreadsheet + * | sql | Returns the generated SQL rather than running the query + * | png | A PNG image of the visualization of the query + * | jpg | A JPG image of the visualization of the query + * + * + * + */ + @Override + public void runUrlEncodedQuery(RunUrlEncodedQueryReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/queries/models/{model_name}/views/{view_name}/run/{result_format}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + RunUrlEncodedQueryResponse.Builder responseBuilder = RunUrlEncodedQueryResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get Merge Query + * + * Returns a merge query object given its id. + * + */ + @Override + public void mergeQuery(MergeQueryReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/merge_queries/{merge_query_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + MergeQueryResponse.Builder responseBuilder = MergeQueryResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create Merge Query + * + * Creates a new merge query object. + * + * A merge query takes the results of one or more queries and combines (merges) the results + * according to field mapping definitions. The result is similar to a SQL left outer join. + * + * A merge query can merge results of queries from different SQL databases. + * + * The order that queries are defined in the source_queries array property is significant. The + * first query in the array defines the primary key into which the results of subsequent + * queries will be merged. + * + * Like model/view query objects, merge queries are immutable and have structural identity - if + * you make a request to create a new merge query that is identical to an existing merge query, + * the existing merge query will be returned instead of creating a duplicate. Conversely, any + * change to the contents of a merge query will produce a new object with a new id. + * + */ + @Override + public void createMergeQuery(CreateMergeQueryReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/merge_queries", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateMergeQueryResponse.Builder responseBuilder = CreateMergeQueryResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * Get information about all running queries. + * + */ + @Override + public void allRunningQueries(AllRunningQueriesReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/running_queries", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllRunningQueriesResponse.Builder responseBuilder = AllRunningQueriesResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + AllRunningQueriesStreamResponse.Builder responseBuilder2 = AllRunningQueriesStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * Kill a query with a specific query_task_id. + * + */ + @Override + public void killQuery(KillQueryReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/running_queries/{query_task_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + KillQueryResponse.Builder responseBuilder = KillQueryResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * Get a SQL Runner query. + */ + @Override + public void sqlQuery(SqlQueryReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/sql_queries/{slug}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SqlQueryResponse.Builder responseBuilder = SqlQueryResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create a SQL Runner Query + * + * Either the `connection_name` or `model_name` parameter MUST be provided. + * + */ + @Override + public void createSqlQuery(CreateSqlQueryReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/sql_queries", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateSqlQueryResponse.Builder responseBuilder = CreateSqlQueryResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * Execute a SQL Runner query in a given result_format. + */ + @Override + public void runSqlQuery(RunSqlQueryReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/sql_queries/{slug}/run/{result_format}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + RunSqlQueryResponse.Builder responseBuilder = RunSqlQueryResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + //#endregion Query: Run and Manage Queries + + //#region RenderTask: Manage Render Tasks + + /** + * ### Create a new task to render a look to an image. + * + * Returns a render task object. + * To check the status of a render task, pass the render_task.id to [Get Render Task](#!/RenderTask/get_render_task). + * Once the render task is complete, you can download the resulting document or image using [Get Render Task Results](#!/RenderTask/get_render_task_results). + * + * + */ + @Override + public void createLookRenderTask(CreateLookRenderTaskReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/render_tasks/looks/{look_id}/{result_format}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateLookRenderTaskResponse.Builder responseBuilder = CreateLookRenderTaskResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create a new task to render an existing query to an image. + * + * Returns a render task object. + * To check the status of a render task, pass the render_task.id to [Get Render Task](#!/RenderTask/get_render_task). + * Once the render task is complete, you can download the resulting document or image using [Get Render Task Results](#!/RenderTask/get_render_task_results). + * + * + */ + @Override + public void createQueryRenderTask(CreateQueryRenderTaskReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/render_tasks/queries/{query_id}/{result_format}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateQueryRenderTaskResponse.Builder responseBuilder = CreateQueryRenderTaskResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create a new task to render a dashboard to a document or image. + * + * Returns a render task object. + * To check the status of a render task, pass the render_task.id to [Get Render Task](#!/RenderTask/get_render_task). + * Once the render task is complete, you can download the resulting document or image using [Get Render Task Results](#!/RenderTask/get_render_task_results). + * + * + */ + @Override + public void createDashboardRenderTask(CreateDashboardRenderTaskReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/render_tasks/dashboards/{dashboard_id}/{result_format}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateDashboardRenderTaskResponse.Builder responseBuilder = CreateDashboardRenderTaskResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about a render task. + * + * Returns a render task object. + * To check the status of a render task, pass the render_task.id to [Get Render Task](#!/RenderTask/get_render_task). + * Once the render task is complete, you can download the resulting document or image using [Get Render Task Results](#!/RenderTask/get_render_task_results). + * + * + */ + @Override + public void renderTask(RenderTaskReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/render_tasks/{render_task_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + RenderTaskResponse.Builder responseBuilder = RenderTaskResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get the document or image produced by a completed render task. + * + * Note that the PDF or image result will be a binary blob in the HTTP response, as indicated by the + * Content-Type in the response headers. This may require specialized (or at least different) handling than text + * responses such as JSON. You may need to tell your HTTP client that the response is binary so that it does not + * attempt to parse the binary data as text. + * + * If the render task exists but has not finished rendering the results, the response HTTP status will be + * **202 Accepted**, the response body will be empty, and the response will have a Retry-After header indicating + * that the caller should repeat the request at a later time. + * + * Returns 404 if the render task cannot be found, if the cached result has expired, or if the caller + * does not have permission to view the results. + * + * For detailed information about the status of the render task, use [Render Task](#!/RenderTask/render_task). + * Polling loops waiting for completion of a render task would be better served by polling **render_task(id)** until + * the task status reaches completion (or error) instead of polling **render_task_results(id)** alone. + * + */ + @Override + public void renderTaskResults(RenderTaskResultsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/render_tasks/{render_task_id}/results", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + RenderTaskResultsResponse.Builder responseBuilder = RenderTaskResultsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create a new task to render a dashboard element to an image. + * + * Returns a render task object. + * To check the status of a render task, pass the render_task.id to [Get Render Task](#!/RenderTask/get_render_task). + * Once the render task is complete, you can download the resulting document or image using [Get Render Task Results](#!/RenderTask/get_render_task_results). + * + * + */ + @Override + public void createDashboardElementRenderTask(CreateDashboardElementRenderTaskReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/render_tasks/dashboard_elements/{dashboard_element_id}/{result_format}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateDashboardElementRenderTaskResponse.Builder responseBuilder = CreateDashboardElementRenderTaskResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + //#endregion RenderTask: Manage Render Tasks + + //#region Role: Manage Roles + + /** + * ### Search model sets + * Returns all model set records that match the given search criteria. + * If multiple search params are given and `filter_or` is FALSE or not specified, + * search params are combined in a logical AND operation. + * Only rows that match *all* search param criteria will be returned. + * + * If `filter_or` is TRUE, multiple search params are combined in a logical OR operation. + * Results will include rows that match **any** of the search criteria. + * + * String search params use case-insensitive matching. + * String search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions. + * example="dan%" will match "danger" and "Danzig" but not "David" + * example="D_m%" will match "Damage" and "dump" + * + * Integer search params can accept a single value or a comma separated list of values. The multiple + * values will be combined under a logical OR operation - results will match at least one of + * the given values. + * + * Most search params can accept "IS NULL" and "NOT NULL" as special expressions to match + * or exclude (respectively) rows where the column is null. + * + * Boolean search params accept only "true" and "false" as values. + * + * + */ + @Override + public void searchModelSets(SearchModelSetsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/model_sets/search", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SearchModelSetsResponse.Builder responseBuilder = SearchModelSetsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + SearchModelSetsStreamResponse.Builder responseBuilder2 = SearchModelSetsStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about the model set with a specific id. + * + */ + @Override + public void modelSet(ModelSetReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/model_sets/{model_set_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ModelSetResponse.Builder responseBuilder = ModelSetResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update information about the model set with a specific id. + * + */ + @Override + public void updateModelSet(UpdateModelSetReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/model_sets/{model_set_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateModelSetResponse.Builder responseBuilder = UpdateModelSetResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Delete the model set with a specific id. + * + */ + @Override + public void deleteModelSet(DeleteModelSetReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/model_sets/{model_set_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteModelSetResponse.Builder responseBuilder = DeleteModelSetResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about all model sets. + * + */ + @Override + public void allModelSets(AllModelSetsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/model_sets", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllModelSetsResponse.Builder responseBuilder = AllModelSetsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + AllModelSetsStreamResponse.Builder responseBuilder2 = AllModelSetsStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create a model set with the specified information. Model sets are used by Roles. + * + */ + @Override + public void createModelSet(CreateModelSetReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/model_sets", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateModelSetResponse.Builder responseBuilder = CreateModelSetResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get all supported permissions. + * + */ + @Override + public void allPermissions(AllPermissionsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/permissions", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllPermissionsResponse.Builder responseBuilder = AllPermissionsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + AllPermissionsStreamResponse.Builder responseBuilder2 = AllPermissionsStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Search permission sets + * Returns all permission set records that match the given search criteria. + * If multiple search params are given and `filter_or` is FALSE or not specified, + * search params are combined in a logical AND operation. + * Only rows that match *all* search param criteria will be returned. + * + * If `filter_or` is TRUE, multiple search params are combined in a logical OR operation. + * Results will include rows that match **any** of the search criteria. + * + * String search params use case-insensitive matching. + * String search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions. + * example="dan%" will match "danger" and "Danzig" but not "David" + * example="D_m%" will match "Damage" and "dump" + * + * Integer search params can accept a single value or a comma separated list of values. The multiple + * values will be combined under a logical OR operation - results will match at least one of + * the given values. + * + * Most search params can accept "IS NULL" and "NOT NULL" as special expressions to match + * or exclude (respectively) rows where the column is null. + * + * Boolean search params accept only "true" and "false" as values. + * + * + */ + @Override + public void searchPermissionSets(SearchPermissionSetsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/permission_sets/search", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SearchPermissionSetsResponse.Builder responseBuilder = SearchPermissionSetsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + SearchPermissionSetsStreamResponse.Builder responseBuilder2 = SearchPermissionSetsStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about the permission set with a specific id. + * + */ + @Override + public void permissionSet(PermissionSetReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/permission_sets/{permission_set_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + PermissionSetResponse.Builder responseBuilder = PermissionSetResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update information about the permission set with a specific id. + * + */ + @Override + public void updatePermissionSet(UpdatePermissionSetReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/permission_sets/{permission_set_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdatePermissionSetResponse.Builder responseBuilder = UpdatePermissionSetResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Delete the permission set with a specific id. + * + */ + @Override + public void deletePermissionSet(DeletePermissionSetReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/permission_sets/{permission_set_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeletePermissionSetResponse.Builder responseBuilder = DeletePermissionSetResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about all permission sets. + * + */ + @Override + public void allPermissionSets(AllPermissionSetsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/permission_sets", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllPermissionSetsResponse.Builder responseBuilder = AllPermissionSetsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + AllPermissionSetsStreamResponse.Builder responseBuilder2 = AllPermissionSetsStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create a permission set with the specified information. Permission sets are used by Roles. + * + */ + @Override + public void createPermissionSet(CreatePermissionSetReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/permission_sets", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreatePermissionSetResponse.Builder responseBuilder = CreatePermissionSetResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about all roles. + * + */ + @Override + public void allRoles(AllRolesReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/roles", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllRolesResponse.Builder responseBuilder = AllRolesResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + AllRolesStreamResponse.Builder responseBuilder2 = AllRolesStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create a role with the specified information. + * + */ + @Override + public void createRole(CreateRoleReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/roles", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateRoleResponse.Builder responseBuilder = CreateRoleResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Search roles + * + * Returns all role records that match the given search criteria. + * + * If multiple search params are given and `filter_or` is FALSE or not specified, + * search params are combined in a logical AND operation. + * Only rows that match *all* search param criteria will be returned. + * + * If `filter_or` is TRUE, multiple search params are combined in a logical OR operation. + * Results will include rows that match **any** of the search criteria. + * + * String search params use case-insensitive matching. + * String search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions. + * example="dan%" will match "danger" and "Danzig" but not "David" + * example="D_m%" will match "Damage" and "dump" + * + * Integer search params can accept a single value or a comma separated list of values. The multiple + * values will be combined under a logical OR operation - results will match at least one of + * the given values. + * + * Most search params can accept "IS NULL" and "NOT NULL" as special expressions to match + * or exclude (respectively) rows where the column is null. + * + * Boolean search params accept only "true" and "false" as values. + * + * + */ + @Override + public void searchRoles(SearchRolesReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/roles/search", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SearchRolesResponse.Builder responseBuilder = SearchRolesResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + SearchRolesStreamResponse.Builder responseBuilder2 = SearchRolesStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Search roles include user count + * + * Returns all role records that match the given search criteria, and attaches + * associated user counts. + * + * If multiple search params are given and `filter_or` is FALSE or not specified, + * search params are combined in a logical AND operation. + * Only rows that match *all* search param criteria will be returned. + * + * If `filter_or` is TRUE, multiple search params are combined in a logical OR operation. + * Results will include rows that match **any** of the search criteria. + * + * String search params use case-insensitive matching. + * String search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions. + * example="dan%" will match "danger" and "Danzig" but not "David" + * example="D_m%" will match "Damage" and "dump" + * + * Integer search params can accept a single value or a comma separated list of values. The multiple + * values will be combined under a logical OR operation - results will match at least one of + * the given values. + * + * Most search params can accept "IS NULL" and "NOT NULL" as special expressions to match + * or exclude (respectively) rows where the column is null. + * + * Boolean search params accept only "true" and "false" as values. + * + * + */ + @Override + public void searchRolesWithUserCount(SearchRolesWithUserCountReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/roles/search/with_user_count", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SearchRolesWithUserCountResponse.Builder responseBuilder = SearchRolesWithUserCountResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + SearchRolesWithUserCountStreamResponse.Builder responseBuilder2 = SearchRolesWithUserCountStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about the role with a specific id. + * + */ + @Override + public void role(RoleReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/roles/{role_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + RoleResponse.Builder responseBuilder = RoleResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update information about the role with a specific id. + * + */ + @Override + public void updateRole(UpdateRoleReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/roles/{role_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateRoleResponse.Builder responseBuilder = UpdateRoleResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Delete the role with a specific id. + * + */ + @Override + public void deleteRole(DeleteRoleReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/roles/{role_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteRoleResponse.Builder responseBuilder = DeleteRoleResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about all the groups with the role that has a specific id. + * + */ + @Override + public void roleGroups(RoleGroupsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/roles/{role_id}/groups", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + RoleGroupsResponse.Builder responseBuilder = RoleGroupsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + RoleGroupsStreamResponse.Builder responseBuilder2 = RoleGroupsStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Set all groups for a role, removing all existing group associations from that role. + * + */ + @Override + public void setRoleGroups(SetRoleGroupsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.put("/roles/{role_id}/groups", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SetRoleGroupsResponse.Builder responseBuilder = SetRoleGroupsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + SetRoleGroupsStreamResponse.Builder responseBuilder2 = SetRoleGroupsStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about all the users with the role that has a specific id. + * + */ + @Override + public void roleUsers(RoleUsersReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/roles/{role_id}/users", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + RoleUsersResponse.Builder responseBuilder = RoleUsersResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + RoleUsersStreamResponse.Builder responseBuilder2 = RoleUsersStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Set all the users of the role with a specific id. + * + */ + @Override + public void setRoleUsers(SetRoleUsersReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.put("/roles/{role_id}/users", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SetRoleUsersResponse.Builder responseBuilder = SetRoleUsersResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + SetRoleUsersStreamResponse.Builder responseBuilder2 = SetRoleUsersStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + //#endregion Role: Manage Roles + + //#region ScheduledPlan: Manage Scheduled Plans + + /** + * ### Get Scheduled Plans for a Space + * + * Returns scheduled plans owned by the caller for a given space id. + * + */ + @Override + public void scheduledPlansForSpace(ScheduledPlansForSpaceReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/scheduled_plans/space/{space_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ScheduledPlansForSpaceResponse.Builder responseBuilder = ScheduledPlansForSpaceResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + ScheduledPlansForSpaceStreamResponse.Builder responseBuilder2 = ScheduledPlansForSpaceStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get Information About a Scheduled Plan + * + * Admins can fetch information about other users' Scheduled Plans. + * + */ + @Override + public void scheduledPlan(ScheduledPlanReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/scheduled_plans/{scheduled_plan_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ScheduledPlanResponse.Builder responseBuilder = ScheduledPlanResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update a Scheduled Plan + * + * Admins can update other users' Scheduled Plans. + * + * Note: Any scheduled plan destinations specified in an update will **replace** all scheduled plan destinations + * currently defined for the scheduled plan. + * + * For Example: If a scheduled plan has destinations A, B, and C, and you call update on this scheduled plan + * specifying only B in the destinations, then destinations A and C will be deleted by the update. + * + * Updating a scheduled plan to assign null or an empty array to the scheduled_plan_destinations property is an error, as a scheduled plan must always have at least one destination. + * + * If you omit the scheduled_plan_destinations property from the object passed to update, then the destinations + * defined on the original scheduled plan will remain unchanged. + * + * #### Email Permissions: + * + * For details about permissions required to schedule delivery to email and the safeguards + * Looker offers to protect against sending to unauthorized email destinations, see [Email Domain Whitelist for Scheduled Looks](https://docs.looker.com/r/api/embed-permissions). + * + * + * #### Scheduled Plan Destination Formats + * + * Scheduled plan destinations must specify the data format to produce and send to the destination. + * + * Formats: + * + * | format | Description + * | :-----------: | :--- | + * | json | A JSON object containing a `data` property which contains an array of JSON objects, one per row. No metadata. + * | json_detail | Row data plus metadata describing the fields, pivots, table calcs, and other aspects of the query + * | inline_json | Same as the JSON format, except that the `data` property is a string containing JSON-escaped row data. Additional properties describe the data operation. This format is primarily used to send data to web hooks so that the web hook doesn't have to re-encode the JSON row data in order to pass it on to its ultimate destination. + * | csv | Comma separated values with a header + * | txt | Tab separated values with a header + * | html | Simple html + * | xlsx | MS Excel spreadsheet + * | wysiwyg_pdf | Dashboard rendered in a tiled layout to produce a PDF document + * | assembled_pdf | Dashboard rendered in a single column layout to produce a PDF document + * | wysiwyg_png | Dashboard rendered in a tiled layout to produce a PNG image + * || + * + * Valid formats vary by destination type and source object. `wysiwyg_pdf` is only valid for dashboards, for example. + * + * + * + */ + @Override + public void updateScheduledPlan(UpdateScheduledPlanReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/scheduled_plans/{scheduled_plan_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateScheduledPlanResponse.Builder responseBuilder = UpdateScheduledPlanResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Delete a Scheduled Plan + * + * Normal users can only delete their own scheduled plans. + * Admins can delete other users' scheduled plans. + * This delete cannot be undone. + * + */ + @Override + public void deleteScheduledPlan(DeleteScheduledPlanReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/scheduled_plans/{scheduled_plan_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteScheduledPlanResponse.Builder responseBuilder = DeleteScheduledPlanResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### List All Scheduled Plans + * + * Returns all scheduled plans which belong to the caller or given user. + * + * If no user_id is provided, this function returns the scheduled plans owned by the caller. + * + * + * To list all schedules for all users, pass `all_users=true`. + * + * + * The caller must have `see_schedules` permission to see other users' scheduled plans. + * + * + * + */ + @Override + public void allScheduledPlans(AllScheduledPlansReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/scheduled_plans", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllScheduledPlansResponse.Builder responseBuilder = AllScheduledPlansResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + AllScheduledPlansStreamResponse.Builder responseBuilder2 = AllScheduledPlansStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create a Scheduled Plan + * + * Create a scheduled plan to render a Look or Dashboard on a recurring schedule. + * + * To create a scheduled plan, you MUST provide values for the following fields: + * `name` + * and + * `look_id`, `dashboard_id`, `lookml_dashboard_id`, or `query_id` + * and + * `cron_tab` or `datagroup` + * and + * at least one scheduled_plan_destination + * + * A scheduled plan MUST have at least one scheduled_plan_destination defined. + * + * When `look_id` is set, `require_no_results`, `require_results`, and `require_change` are all required. + * + * If `create_scheduled_plan` fails with a 422 error, be sure to look at the error messages in the response which will explain exactly what fields are missing or values that are incompatible. + * + * The queries that provide the data for the look or dashboard are run in the context of user account that owns the scheduled plan. + * + * When `run_as_recipient` is `false` or not specified, the queries that provide the data for the + * look or dashboard are run in the context of user account that owns the scheduled plan. + * + * When `run_as_recipient` is `true` and all the email recipients are Looker user accounts, the + * queries are run in the context of each recipient, so different recipients may see different + * data from the same scheduled render of a look or dashboard. For more details, see [Run As Recipient](https://looker.com/docs/r/admin/run-as-recipient). + * + * Admins can create and modify scheduled plans on behalf of other users by specifying a user id. + * Non-admin users may not create or modify scheduled plans by or for other users. + * + * #### Email Permissions: + * + * For details about permissions required to schedule delivery to email and the safeguards + * Looker offers to protect against sending to unauthorized email destinations, see [Email Domain Whitelist for Scheduled Looks](https://docs.looker.com/r/api/embed-permissions). + * + * + * #### Scheduled Plan Destination Formats + * + * Scheduled plan destinations must specify the data format to produce and send to the destination. + * + * Formats: + * + * | format | Description + * | :-----------: | :--- | + * | json | A JSON object containing a `data` property which contains an array of JSON objects, one per row. No metadata. + * | json_detail | Row data plus metadata describing the fields, pivots, table calcs, and other aspects of the query + * | inline_json | Same as the JSON format, except that the `data` property is a string containing JSON-escaped row data. Additional properties describe the data operation. This format is primarily used to send data to web hooks so that the web hook doesn't have to re-encode the JSON row data in order to pass it on to its ultimate destination. + * | csv | Comma separated values with a header + * | txt | Tab separated values with a header + * | html | Simple html + * | xlsx | MS Excel spreadsheet + * | wysiwyg_pdf | Dashboard rendered in a tiled layout to produce a PDF document + * | assembled_pdf | Dashboard rendered in a single column layout to produce a PDF document + * | wysiwyg_png | Dashboard rendered in a tiled layout to produce a PNG image + * || + * + * Valid formats vary by destination type and source object. `wysiwyg_pdf` is only valid for dashboards, for example. + * + * + * + */ + @Override + public void createScheduledPlan(CreateScheduledPlanReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/scheduled_plans", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateScheduledPlanResponse.Builder responseBuilder = CreateScheduledPlanResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Run a Scheduled Plan Immediately + * + * Create a scheduled plan that runs only once, and immediately. + * + * This can be useful for testing a Scheduled Plan before committing to a production schedule. + * + * Admins can create scheduled plans on behalf of other users by specifying a user id. + * + * This API is rate limited to prevent it from being used for relay spam or DoS attacks + * + * #### Email Permissions: + * + * For details about permissions required to schedule delivery to email and the safeguards + * Looker offers to protect against sending to unauthorized email destinations, see [Email Domain Whitelist for Scheduled Looks](https://docs.looker.com/r/api/embed-permissions). + * + * + * #### Scheduled Plan Destination Formats + * + * Scheduled plan destinations must specify the data format to produce and send to the destination. + * + * Formats: + * + * | format | Description + * | :-----------: | :--- | + * | json | A JSON object containing a `data` property which contains an array of JSON objects, one per row. No metadata. + * | json_detail | Row data plus metadata describing the fields, pivots, table calcs, and other aspects of the query + * | inline_json | Same as the JSON format, except that the `data` property is a string containing JSON-escaped row data. Additional properties describe the data operation. This format is primarily used to send data to web hooks so that the web hook doesn't have to re-encode the JSON row data in order to pass it on to its ultimate destination. + * | csv | Comma separated values with a header + * | txt | Tab separated values with a header + * | html | Simple html + * | xlsx | MS Excel spreadsheet + * | wysiwyg_pdf | Dashboard rendered in a tiled layout to produce a PDF document + * | assembled_pdf | Dashboard rendered in a single column layout to produce a PDF document + * | wysiwyg_png | Dashboard rendered in a tiled layout to produce a PNG image + * || + * + * Valid formats vary by destination type and source object. `wysiwyg_pdf` is only valid for dashboards, for example. + * + * + * + */ + @Override + public void scheduledPlanRunOnce(ScheduledPlanRunOnceReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/scheduled_plans/run_once", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ScheduledPlanRunOnceResponse.Builder responseBuilder = ScheduledPlanRunOnceResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get Scheduled Plans for a Look + * + * Returns all scheduled plans for a look which belong to the caller or given user. + * + * If no user_id is provided, this function returns the scheduled plans owned by the caller. + * + * + * To list all schedules for all users, pass `all_users=true`. + * + * + * The caller must have `see_schedules` permission to see other users' scheduled plans. + * + * + * + */ + @Override + public void scheduledPlansForLook(ScheduledPlansForLookReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/scheduled_plans/look/{look_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ScheduledPlansForLookResponse.Builder responseBuilder = ScheduledPlansForLookResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + ScheduledPlansForLookStreamResponse.Builder responseBuilder2 = ScheduledPlansForLookStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get Scheduled Plans for a Dashboard + * + * Returns all scheduled plans for a dashboard which belong to the caller or given user. + * + * If no user_id is provided, this function returns the scheduled plans owned by the caller. + * + * + * To list all schedules for all users, pass `all_users=true`. + * + * + * The caller must have `see_schedules` permission to see other users' scheduled plans. + * + * + * + */ + @Override + public void scheduledPlansForDashboard(ScheduledPlansForDashboardReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/scheduled_plans/dashboard/{dashboard_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ScheduledPlansForDashboardResponse.Builder responseBuilder = ScheduledPlansForDashboardResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + ScheduledPlansForDashboardStreamResponse.Builder responseBuilder2 = ScheduledPlansForDashboardStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get Scheduled Plans for a LookML Dashboard + * + * Returns all scheduled plans for a LookML Dashboard which belong to the caller or given user. + * + * If no user_id is provided, this function returns the scheduled plans owned by the caller. + * + * + * To list all schedules for all users, pass `all_users=true`. + * + * + * The caller must have `see_schedules` permission to see other users' scheduled plans. + * + * + * + */ + @Override + public void scheduledPlansForLookmlDashboard(ScheduledPlansForLookmlDashboardReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/scheduled_plans/lookml_dashboard/{lookml_dashboard_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ScheduledPlansForLookmlDashboardResponse.Builder responseBuilder = ScheduledPlansForLookmlDashboardResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + ScheduledPlansForLookmlDashboardStreamResponse.Builder responseBuilder2 = ScheduledPlansForLookmlDashboardStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Run a Scheduled Plan By Id Immediately + * This function creates a run-once schedule plan based on an existing scheduled plan, + * applies modifications (if any) to the new scheduled plan, and runs the new schedule plan immediately. + * This can be useful for testing modifications to an existing scheduled plan before committing to a production schedule. + * + * This function internally performs the following operations: + * + * 1. Copies the properties of the existing scheduled plan into a new scheduled plan + * 2. Copies any properties passed in the JSON body of this request into the new scheduled plan (replacing the original values) + * 3. Creates the new scheduled plan + * 4. Runs the new scheduled plan + * + * The original scheduled plan is not modified by this operation. + * Admins can create, modify, and run scheduled plans on behalf of other users by specifying a user id. + * Non-admins can only create, modify, and run their own scheduled plans. + * + * #### Email Permissions: + * + * For details about permissions required to schedule delivery to email and the safeguards + * Looker offers to protect against sending to unauthorized email destinations, see [Email Domain Whitelist for Scheduled Looks](https://docs.looker.com/r/api/embed-permissions). + * + * + * #### Scheduled Plan Destination Formats + * + * Scheduled plan destinations must specify the data format to produce and send to the destination. + * + * Formats: + * + * | format | Description + * | :-----------: | :--- | + * | json | A JSON object containing a `data` property which contains an array of JSON objects, one per row. No metadata. + * | json_detail | Row data plus metadata describing the fields, pivots, table calcs, and other aspects of the query + * | inline_json | Same as the JSON format, except that the `data` property is a string containing JSON-escaped row data. Additional properties describe the data operation. This format is primarily used to send data to web hooks so that the web hook doesn't have to re-encode the JSON row data in order to pass it on to its ultimate destination. + * | csv | Comma separated values with a header + * | txt | Tab separated values with a header + * | html | Simple html + * | xlsx | MS Excel spreadsheet + * | wysiwyg_pdf | Dashboard rendered in a tiled layout to produce a PDF document + * | assembled_pdf | Dashboard rendered in a single column layout to produce a PDF document + * | wysiwyg_png | Dashboard rendered in a tiled layout to produce a PNG image + * || + * + * Valid formats vary by destination type and source object. `wysiwyg_pdf` is only valid for dashboards, for example. + * + * + * + * This API is rate limited to prevent it from being used for relay spam or DoS attacks + * + * + */ + @Override + public void scheduledPlanRunOnceById(ScheduledPlanRunOnceByIdReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/scheduled_plans/{scheduled_plan_id}/run_once", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ScheduledPlanRunOnceByIdResponse.Builder responseBuilder = ScheduledPlanRunOnceByIdResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + //#endregion ScheduledPlan: Manage Scheduled Plans + + //#region Session: Session Information + + /** + * ### Get API Session + * + * Returns information about the current API session, such as which workspace is selected for the session. + * + */ + @Override + public void session(SessionReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/session", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SessionResponse.Builder responseBuilder = SessionResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update API Session + * + * #### API Session Workspace + * + * You can use this endpoint to change the active workspace for the current API session. + * + * Only one workspace can be active in a session. The active workspace can be changed + * any number of times in a session. + * + * The default workspace for API sessions is the "production" workspace. + * + * All Looker APIs that use projects or lookml models (such as running queries) will + * use the version of project and model files defined by this workspace for the lifetime of the + * current API session or until the session workspace is changed again. + * + * An API session has the same lifetime as the access_token used to authenticate API requests. Each successful + * API login generates a new access_token and a new API session. + * + * If your Looker API client application needs to work in a dev workspace across multiple + * API sessions, be sure to select the dev workspace after each login. + * + */ + @Override + public void updateSession(UpdateSessionReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/session", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateSessionResponse.Builder responseBuilder = UpdateSessionResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + //#endregion Session: Session Information + + //#region Theme: Manage Themes + + /** + * ### Get an array of all existing themes + * + * Get a **single theme** by id with [Theme](#!/Theme/theme) + * + * This method returns an array of all existing themes. The active time for the theme is not considered. + * + * **Note**: Custom themes needs to be enabled by Looker. Unless custom themes are enabled, only the automatically generated default theme can be used. Please contact your Account Manager or help.looker.com to update your license for this feature. + * + * + */ + @Override + public void allThemes(AllThemesReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/themes", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllThemesResponse.Builder responseBuilder = AllThemesResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + AllThemesStreamResponse.Builder responseBuilder2 = AllThemesStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create a theme + * + * Creates a new theme object, returning the theme details, including the created id. + * + * If `settings` are not specified, the default theme settings will be copied into the new theme. + * + * The theme `name` can only contain alphanumeric characters or underscores. Theme names should not contain any confidential information, such as customer names. + * + * **Update** an existing theme with [Update Theme](#!/Theme/update_theme) + * + * **Permanently delete** an existing theme with [Delete Theme](#!/Theme/delete_theme) + * + * For more information, see [Creating and Applying Themes](https://looker.com/docs/r/admin/themes). + * + * **Note**: Custom themes needs to be enabled by Looker. Unless custom themes are enabled, only the automatically generated default theme can be used. Please contact your Account Manager or help.looker.com to update your license for this feature. + * + * + */ + @Override + public void createTheme(CreateThemeReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/themes", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateThemeResponse.Builder responseBuilder = CreateThemeResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Search all themes for matching criteria. + * + * Returns an **array of theme objects** that match the specified search criteria. + * + * | Search Parameters | Description + * | :-------------------: | :------ | + * | `begin_at` only | Find themes active at or after `begin_at` + * | `end_at` only | Find themes active at or before `end_at` + * | both set | Find themes with an active inclusive period between `begin_at` and `end_at` + * + * Note: Range matching requires boolean AND logic. + * When using `begin_at` and `end_at` together, do not use `filter_or`=TRUE + * + * If multiple search params are given and `filter_or` is FALSE or not specified, + * search params are combined in a logical AND operation. + * Only rows that match *all* search param criteria will be returned. + * + * If `filter_or` is TRUE, multiple search params are combined in a logical OR operation. + * Results will include rows that match **any** of the search criteria. + * + * String search params use case-insensitive matching. + * String search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions. + * example="dan%" will match "danger" and "Danzig" but not "David" + * example="D_m%" will match "Damage" and "dump" + * + * Integer search params can accept a single value or a comma separated list of values. The multiple + * values will be combined under a logical OR operation - results will match at least one of + * the given values. + * + * Most search params can accept "IS NULL" and "NOT NULL" as special expressions to match + * or exclude (respectively) rows where the column is null. + * + * Boolean search params accept only "true" and "false" as values. + * + * + * Get a **single theme** by id with [Theme](#!/Theme/theme) + * + * **Note**: Custom themes needs to be enabled by Looker. Unless custom themes are enabled, only the automatically generated default theme can be used. Please contact your Account Manager or help.looker.com to update your license for this feature. + * + * + */ + @Override + public void searchThemes(SearchThemesReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/themes/search", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SearchThemesResponse.Builder responseBuilder = SearchThemesResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + SearchThemesStreamResponse.Builder responseBuilder2 = SearchThemesStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get the default theme + * + * Returns the active theme object set as the default. + * + * The **default** theme name can be set in the UI on the Admin|Theme UI page + * + * The optional `ts` parameter can specify a different timestamp than "now." If specified, it returns the default theme at the time indicated. + * + */ + @Override + public void defaultTheme(DefaultThemeReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/themes/default", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DefaultThemeResponse.Builder responseBuilder = DefaultThemeResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Set the global default theme by theme name + * + * Only Admin users can call this function. + * + * Only an active theme with no expiration (`end_at` not set) can be assigned as the default theme. As long as a theme has an active record with no expiration, it can be set as the default. + * + * [Create Theme](#!/Theme/create) has detailed information on rules for default and active themes + * + * Returns the new specified default theme object. + * + * **Note**: Custom themes needs to be enabled by Looker. Unless custom themes are enabled, only the automatically generated default theme can be used. Please contact your Account Manager or help.looker.com to update your license for this feature. + * + * + */ + @Override + public void setDefaultTheme(SetDefaultThemeReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.put("/themes/default", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SetDefaultThemeResponse.Builder responseBuilder = SetDefaultThemeResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get active themes + * + * Returns an array of active themes. + * + * If the `name` parameter is specified, it will return an array with one theme if it's active and found. + * + * The optional `ts` parameter can specify a different timestamp than "now." + * + * **Note**: Custom themes needs to be enabled by Looker. Unless custom themes are enabled, only the automatically generated default theme can be used. Please contact your Account Manager or help.looker.com to update your license for this feature. + * + * + * + */ + @Override + public void activeThemes(ActiveThemesReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/themes/active", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ActiveThemesResponse.Builder responseBuilder = ActiveThemesResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + ActiveThemesStreamResponse.Builder responseBuilder2 = ActiveThemesStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get the named theme if it's active. Otherwise, return the default theme + * + * The optional `ts` parameter can specify a different timestamp than "now." + * Note: API users with `show` ability can call this function + * + * **Note**: Custom themes needs to be enabled by Looker. Unless custom themes are enabled, only the automatically generated default theme can be used. Please contact your Account Manager or help.looker.com to update your license for this feature. + * + * + */ + @Override + public void themeOrDefault(ThemeOrDefaultReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/themes/theme_or_default", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ThemeOrDefaultResponse.Builder responseBuilder = ThemeOrDefaultResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Validate a theme with the specified information + * + * Validates all values set for the theme, returning any errors encountered, or 200 OK if valid + * + * See [Create Theme](#!/Theme/create_theme) for constraints + * + * **Note**: Custom themes needs to be enabled by Looker. Unless custom themes are enabled, only the automatically generated default theme can be used. Please contact your Account Manager or help.looker.com to update your license for this feature. + * + * + */ + @Override + public void validateTheme(ValidateThemeReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/themes/validate", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ValidateThemeResponse.Builder responseBuilder = ValidateThemeResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get a theme by ID + * + * Use this to retrieve a specific theme, whether or not it's currently active. + * + * **Note**: Custom themes needs to be enabled by Looker. Unless custom themes are enabled, only the automatically generated default theme can be used. Please contact your Account Manager or help.looker.com to update your license for this feature. + * + * + */ + @Override + public void theme(ThemeReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/themes/{theme_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + ThemeResponse.Builder responseBuilder = ThemeResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update the theme by id. + * + * **Note**: Custom themes needs to be enabled by Looker. Unless custom themes are enabled, only the automatically generated default theme can be used. Please contact your Account Manager or help.looker.com to update your license for this feature. + * + * + */ + @Override + public void updateTheme(UpdateThemeReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/themes/{theme_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateThemeResponse.Builder responseBuilder = UpdateThemeResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Delete a specific theme by id + * + * This operation permanently deletes the identified theme from the database. + * + * Because multiple themes can have the same name (with different activation time spans) themes can only be deleted by ID. + * + * All IDs associated with a theme name can be retrieved by searching for the theme name with [Theme Search](#!/Theme/search). + * + * **Note**: Custom themes needs to be enabled by Looker. Unless custom themes are enabled, only the automatically generated default theme can be used. Please contact your Account Manager or help.looker.com to update your license for this feature. + * + * + */ + @Override + public void deleteTheme(DeleteThemeReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/themes/{theme_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteThemeResponse.Builder responseBuilder = DeleteThemeResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + //#endregion Theme: Manage Themes + + //#region User: Manage Users + + /** + * ### Search email credentials + * + * Returns all credentials_email records that match the given search criteria. + * + * If multiple search params are given and `filter_or` is FALSE or not specified, + * search params are combined in a logical AND operation. + * Only rows that match *all* search param criteria will be returned. + * + * If `filter_or` is TRUE, multiple search params are combined in a logical OR operation. + * Results will include rows that match **any** of the search criteria. + * + * String search params use case-insensitive matching. + * String search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions. + * example="dan%" will match "danger" and "Danzig" but not "David" + * example="D_m%" will match "Damage" and "dump" + * + * Integer search params can accept a single value or a comma separated list of values. The multiple + * values will be combined under a logical OR operation - results will match at least one of + * the given values. + * + * Most search params can accept "IS NULL" and "NOT NULL" as special expressions to match + * or exclude (respectively) rows where the column is null. + * + * Boolean search params accept only "true" and "false" as values. + * + * + */ + @Override + public void searchCredentialsEmail(SearchCredentialsEmailReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/credentials_email/search", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SearchCredentialsEmailResponse.Builder responseBuilder = SearchCredentialsEmailResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + SearchCredentialsEmailStreamResponse.Builder responseBuilder2 = SearchCredentialsEmailStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about the current user; i.e. the user account currently calling the API. + * + */ + @Override + public void me(MeReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/user", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + MeResponse.Builder responseBuilder = MeResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about all users. + * + */ + @Override + public void allUsers(AllUsersReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/users", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllUsersResponse.Builder responseBuilder = AllUsersResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + AllUsersStreamResponse.Builder responseBuilder2 = AllUsersStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create a user with the specified information. + * + */ + @Override + public void createUser(CreateUserReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/users", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateUserResponse.Builder responseBuilder = CreateUserResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Search users + * + * Returns all* user records that match the given search criteria. + * + * If multiple search params are given and `filter_or` is FALSE or not specified, + * search params are combined in a logical AND operation. + * Only rows that match *all* search param criteria will be returned. + * + * If `filter_or` is TRUE, multiple search params are combined in a logical OR operation. + * Results will include rows that match **any** of the search criteria. + * + * String search params use case-insensitive matching. + * String search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions. + * example="dan%" will match "danger" and "Danzig" but not "David" + * example="D_m%" will match "Damage" and "dump" + * + * Integer search params can accept a single value or a comma separated list of values. The multiple + * values will be combined under a logical OR operation - results will match at least one of + * the given values. + * + * Most search params can accept "IS NULL" and "NOT NULL" as special expressions to match + * or exclude (respectively) rows where the column is null. + * + * Boolean search params accept only "true" and "false" as values. + * + * + * (*) Results are always filtered to the level of information the caller is permitted to view. + * Looker admins can see all user details; normal users in an open system can see + * names of other users but no details; normal users in a closed system can only see + * names of other users who are members of the same group as the user. + * + * + */ + @Override + public void searchUsers(SearchUsersReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/users/search", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SearchUsersResponse.Builder responseBuilder = SearchUsersResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + SearchUsersStreamResponse.Builder responseBuilder2 = SearchUsersStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Search for user accounts by name + * + * Returns all user accounts where `first_name` OR `last_name` OR `email` field values match a pattern. + * The pattern can contain `%` and `_` wildcards as in SQL LIKE expressions. + * + * Any additional search params will be combined into a logical AND expression. + * + */ + @Override + public void searchUsersNames(SearchUsersNamesReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/users/search/names/{pattern}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SearchUsersNamesResponse.Builder responseBuilder = SearchUsersNamesResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + SearchUsersNamesStreamResponse.Builder responseBuilder2 = SearchUsersNamesStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about the user with a specific id. + * + * If the caller is an admin or the caller is the user being specified, then full user information will + * be returned. Otherwise, a minimal 'public' variant of the user information will be returned. This contains + * The user name and avatar url, but no sensitive information. + * + */ + @Override + public void user(UserReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/users/{user_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UserResponse.Builder responseBuilder = UserResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update information about the user with a specific id. + * + */ + @Override + public void updateUser(UpdateUserReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/users/{user_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateUserResponse.Builder responseBuilder = UpdateUserResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Delete the user with a specific id. + * + * **DANGER** this will delete the user and all looks and other information owned by the user. + * + */ + @Override + public void deleteUser(DeleteUserReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/users/{user_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteUserResponse.Builder responseBuilder = DeleteUserResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about the user with a credential of given type with specific id. + * + * This is used to do things like find users by their embed external_user_id. Or, find the user with + * a given api3 client_id, etc. The 'credential_type' matches the 'type' name of the various credential + * types. It must be one of the values listed in the table below. The 'credential_id' is your unique Id + * for the user and is specific to each type of credential. + * + * An example using the Ruby sdk might look like: + * + * `sdk.user_for_credential('embed', 'customer-4959425')` + * + * This table shows the supported 'Credential Type' strings. The right column is for reference; it shows + * which field in the given credential type is actually searched when finding a user with the supplied + * 'credential_id'. + * + * | Credential Types | Id Field Matched | + * | ---------------- | ---------------- | + * | email | email | + * | google | google_user_id | + * | saml | saml_user_id | + * | oidc | oidc_user_id | + * | ldap | ldap_id | + * | api | token | + * | api3 | client_id | + * | embed | external_user_id | + * | looker_openid | email | + * + * **NOTE**: The 'api' credential type was only used with the legacy Looker query API and is no longer supported. The credential type for API you are currently looking at is 'api3'. + * + * + */ + @Override + public void userForCredential(UserForCredentialReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/users/credential/{credential_type}/{credential_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UserForCredentialResponse.Builder responseBuilder = UserForCredentialResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Email/password login information for the specified user. + */ + @Override + public void userCredentialsEmail(UserCredentialsEmailReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/users/{user_id}/credentials_email", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UserCredentialsEmailResponse.Builder responseBuilder = UserCredentialsEmailResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Email/password login information for the specified user. + */ + @Override + public void createUserCredentialsEmail(CreateUserCredentialsEmailReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/users/{user_id}/credentials_email", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateUserCredentialsEmailResponse.Builder responseBuilder = CreateUserCredentialsEmailResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Email/password login information for the specified user. + */ + @Override + public void updateUserCredentialsEmail(UpdateUserCredentialsEmailReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/users/{user_id}/credentials_email", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateUserCredentialsEmailResponse.Builder responseBuilder = UpdateUserCredentialsEmailResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Email/password login information for the specified user. + */ + @Override + public void deleteUserCredentialsEmail(DeleteUserCredentialsEmailReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/users/{user_id}/credentials_email", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteUserCredentialsEmailResponse.Builder responseBuilder = DeleteUserCredentialsEmailResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Two-factor login information for the specified user. + */ + @Override + public void userCredentialsTotp(UserCredentialsTotpReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/users/{user_id}/credentials_totp", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UserCredentialsTotpResponse.Builder responseBuilder = UserCredentialsTotpResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Two-factor login information for the specified user. + */ + @Override + public void createUserCredentialsTotp(CreateUserCredentialsTotpReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/users/{user_id}/credentials_totp", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateUserCredentialsTotpResponse.Builder responseBuilder = CreateUserCredentialsTotpResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Two-factor login information for the specified user. + */ + @Override + public void deleteUserCredentialsTotp(DeleteUserCredentialsTotpReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/users/{user_id}/credentials_totp", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteUserCredentialsTotpResponse.Builder responseBuilder = DeleteUserCredentialsTotpResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### LDAP login information for the specified user. + */ + @Override + public void userCredentialsLdap(UserCredentialsLdapReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/users/{user_id}/credentials_ldap", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UserCredentialsLdapResponse.Builder responseBuilder = UserCredentialsLdapResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### LDAP login information for the specified user. + */ + @Override + public void deleteUserCredentialsLdap(DeleteUserCredentialsLdapReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/users/{user_id}/credentials_ldap", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteUserCredentialsLdapResponse.Builder responseBuilder = DeleteUserCredentialsLdapResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Google authentication login information for the specified user. + */ + @Override + public void userCredentialsGoogle(UserCredentialsGoogleReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/users/{user_id}/credentials_google", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UserCredentialsGoogleResponse.Builder responseBuilder = UserCredentialsGoogleResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Google authentication login information for the specified user. + */ + @Override + public void deleteUserCredentialsGoogle(DeleteUserCredentialsGoogleReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/users/{user_id}/credentials_google", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteUserCredentialsGoogleResponse.Builder responseBuilder = DeleteUserCredentialsGoogleResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Saml authentication login information for the specified user. + */ + @Override + public void userCredentialsSaml(UserCredentialsSamlReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/users/{user_id}/credentials_saml", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UserCredentialsSamlResponse.Builder responseBuilder = UserCredentialsSamlResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Saml authentication login information for the specified user. + */ + @Override + public void deleteUserCredentialsSaml(DeleteUserCredentialsSamlReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/users/{user_id}/credentials_saml", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteUserCredentialsSamlResponse.Builder responseBuilder = DeleteUserCredentialsSamlResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### OpenID Connect (OIDC) authentication login information for the specified user. + */ + @Override + public void userCredentialsOidc(UserCredentialsOidcReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/users/{user_id}/credentials_oidc", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UserCredentialsOidcResponse.Builder responseBuilder = UserCredentialsOidcResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### OpenID Connect (OIDC) authentication login information for the specified user. + */ + @Override + public void deleteUserCredentialsOidc(DeleteUserCredentialsOidcReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/users/{user_id}/credentials_oidc", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteUserCredentialsOidcResponse.Builder responseBuilder = DeleteUserCredentialsOidcResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### API 3 login information for the specified user. This is for the newer API keys that can be added for any user. + */ + @Override + public void userCredentialsApi3(UserCredentialsApi3Req request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/users/{user_id}/credentials_api3/{credentials_api3_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UserCredentialsApi3Response.Builder responseBuilder = UserCredentialsApi3Response.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### API 3 login information for the specified user. This is for the newer API keys that can be added for any user. + */ + @Override + public void deleteUserCredentialsApi3(DeleteUserCredentialsApi3Req request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/users/{user_id}/credentials_api3/{credentials_api3_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteUserCredentialsApi3Response.Builder responseBuilder = DeleteUserCredentialsApi3Response.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### API 3 login information for the specified user. This is for the newer API keys that can be added for any user. + */ + @Override + public void allUserCredentialsApi3s(AllUserCredentialsApi3sReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/users/{user_id}/credentials_api3", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllUserCredentialsApi3sResponse.Builder responseBuilder = AllUserCredentialsApi3sResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + AllUserCredentialsApi3sStreamResponse.Builder responseBuilder2 = AllUserCredentialsApi3sStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### API 3 login information for the specified user. This is for the newer API keys that can be added for any user. + */ + @Override + public void createUserCredentialsApi3(CreateUserCredentialsApi3Req request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/users/{user_id}/credentials_api3", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateUserCredentialsApi3Response.Builder responseBuilder = CreateUserCredentialsApi3Response.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Embed login information for the specified user. + */ + @Override + public void userCredentialsEmbed(UserCredentialsEmbedReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/users/{user_id}/credentials_embed/{credentials_embed_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UserCredentialsEmbedResponse.Builder responseBuilder = UserCredentialsEmbedResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Embed login information for the specified user. + */ + @Override + public void deleteUserCredentialsEmbed(DeleteUserCredentialsEmbedReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/users/{user_id}/credentials_embed/{credentials_embed_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteUserCredentialsEmbedResponse.Builder responseBuilder = DeleteUserCredentialsEmbedResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Embed login information for the specified user. + */ + @Override + public void allUserCredentialsEmbeds(AllUserCredentialsEmbedsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/users/{user_id}/credentials_embed", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllUserCredentialsEmbedsResponse.Builder responseBuilder = AllUserCredentialsEmbedsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + AllUserCredentialsEmbedsStreamResponse.Builder responseBuilder2 = AllUserCredentialsEmbedsStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Looker Openid login information for the specified user. Used by Looker Analysts. + */ + @Override + public void userCredentialsLookerOpenid(UserCredentialsLookerOpenidReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/users/{user_id}/credentials_looker_openid", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UserCredentialsLookerOpenidResponse.Builder responseBuilder = UserCredentialsLookerOpenidResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Looker Openid login information for the specified user. Used by Looker Analysts. + */ + @Override + public void deleteUserCredentialsLookerOpenid(DeleteUserCredentialsLookerOpenidReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/users/{user_id}/credentials_looker_openid", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteUserCredentialsLookerOpenidResponse.Builder responseBuilder = DeleteUserCredentialsLookerOpenidResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Web login session for the specified user. + */ + @Override + public void userSession(UserSessionReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/users/{user_id}/sessions/{session_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UserSessionResponse.Builder responseBuilder = UserSessionResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Web login session for the specified user. + */ + @Override + public void deleteUserSession(DeleteUserSessionReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/users/{user_id}/sessions/{session_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteUserSessionResponse.Builder responseBuilder = DeleteUserSessionResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Web login session for the specified user. + */ + @Override + public void allUserSessions(AllUserSessionsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/users/{user_id}/sessions", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllUserSessionsResponse.Builder responseBuilder = AllUserSessionsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + AllUserSessionsStreamResponse.Builder responseBuilder2 = AllUserSessionsStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create a password reset token. + * This will create a cryptographically secure random password reset token for the user. + * If the user already has a password reset token then this invalidates the old token and creates a new one. + * The token is expressed as the 'password_reset_url' of the user's email/password credential object. + * This takes an optional 'expires' param to indicate if the new token should be an expiring token. + * Tokens that expire are typically used for self-service password resets for existing users. + * Invitation emails for new users typically are not set to expire. + * The expire period is always 60 minutes when expires is enabled. + * This method can be called with an empty body. + * + */ + @Override + public void createUserCredentialsEmailPasswordReset(CreateUserCredentialsEmailPasswordResetReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/users/{user_id}/credentials_email/password_reset", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateUserCredentialsEmailPasswordResetResponse.Builder responseBuilder = CreateUserCredentialsEmailPasswordResetResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about roles of a given user + * + */ + @Override + public void userRoles(UserRolesReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/users/{user_id}/roles", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UserRolesResponse.Builder responseBuilder = UserRolesResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + UserRolesStreamResponse.Builder responseBuilder2 = UserRolesStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Set roles of the user with a specific id. + * + */ + @Override + public void setUserRoles(SetUserRolesReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.put("/users/{user_id}/roles", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SetUserRolesResponse.Builder responseBuilder = SetUserRolesResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + SetUserRolesStreamResponse.Builder responseBuilder2 = SetUserRolesStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get user attribute values for a given user. + * + * Returns the values of specified user attributes (or all user attributes) for a certain user. + * + * A value for each user attribute is searched for in the following locations, in this order: + * + * 1. in the user's account information + * 1. in groups that the user is a member of + * 1. the default value of the user attribute + * + * If more than one group has a value defined for a user attribute, the group with the lowest rank wins. + * + * The response will only include user attributes for which values were found. Use `include_unset=true` to include + * empty records for user attributes with no value. + * + * The value of all hidden user attributes will be blank. + * + */ + @Override + public void userAttributeUserValues(UserAttributeUserValuesReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/users/{user_id}/attribute_values", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UserAttributeUserValuesResponse.Builder responseBuilder = UserAttributeUserValuesResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + UserAttributeUserValuesStreamResponse.Builder responseBuilder2 = UserAttributeUserValuesStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Store a custom value for a user attribute in a user's account settings. + * + * Per-user user attribute values take precedence over group or default values. + * + */ + @Override + public void setUserAttributeUserValue(SetUserAttributeUserValueReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/users/{user_id}/attribute_values/{user_attribute_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SetUserAttributeUserValueResponse.Builder responseBuilder = SetUserAttributeUserValueResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Delete a user attribute value from a user's account settings. + * + * After the user attribute value is deleted from the user's account settings, subsequent requests + * for the user attribute value for this user will draw from the user's groups or the default + * value of the user attribute. See [Get User Attribute Values](#!/User/user_attribute_user_values) for more + * information about how user attribute values are resolved. + * + */ + @Override + public void deleteUserAttributeUserValue(DeleteUserAttributeUserValueReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/users/{user_id}/attribute_values/{user_attribute_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteUserAttributeUserValueResponse.Builder responseBuilder = DeleteUserAttributeUserValueResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Send a password reset token. + * This will send a password reset email to the user. If a password reset token does not already exist + * for this user, it will create one and then send it. + * If the user has not yet set up their account, it will send a setup email to the user. + * The URL sent in the email is expressed as the 'password_reset_url' of the user's email/password credential object. + * Password reset URLs will expire in 60 minutes. + * This method can be called with an empty body. + * + */ + @Override + public void sendUserCredentialsEmailPasswordReset(SendUserCredentialsEmailPasswordResetReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/users/{user_id}/credentials_email/send_password_reset", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SendUserCredentialsEmailPasswordResetResponse.Builder responseBuilder = SendUserCredentialsEmailPasswordResetResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Change a disabled user's email addresses + * + * Allows the admin to change the email addresses for all the user's + * associated credentials. Will overwrite all associated email addresses with + * the value supplied in the 'email' body param. + * The user's 'is_disabled' status must be true. + * + */ + @Override + public void wipeoutUserEmails(WipeoutUserEmailsReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/users/{user_id}/update_emails", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + WipeoutUserEmailsResponse.Builder responseBuilder = WipeoutUserEmailsResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * Create an embed user from an external user ID + * + */ + @Override + public void createEmbedUser(CreateEmbedUserReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/users/embed_user", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateEmbedUserResponse.Builder responseBuilder = CreateEmbedUserResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + //#endregion User: Manage Users + + //#region UserAttribute: Manage User Attributes + + /** + * ### Get information about all user attributes. + * + */ + @Override + public void allUserAttributes(AllUserAttributesReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/user_attributes", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllUserAttributesResponse.Builder responseBuilder = AllUserAttributesResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + AllUserAttributesStreamResponse.Builder responseBuilder2 = AllUserAttributesStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Create a new user attribute + * + * Permission information for a user attribute is conveyed through the `can` and `user_can_edit` fields. + * The `user_can_edit` field indicates whether an attribute is user-editable _anywhere_ in the application. + * The `can` field gives more granular access information, with the `set_value` child field indicating whether + * an attribute's value can be set by [Setting the User Attribute User Value](#!/User/set_user_attribute_user_value). + * + * Note: `name` and `label` fields must be unique across all user attributes in the Looker instance. + * Attempting to create a new user attribute with a name or label that duplicates an existing + * user attribute will fail with a 422 error. + * + */ + @Override + public void createUserAttribute(CreateUserAttributeReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/user_attributes", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + CreateUserAttributeResponse.Builder responseBuilder = CreateUserAttributeResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get information about a user attribute. + * + */ + @Override + public void userAttribute(UserAttributeReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/user_attributes/{user_attribute_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UserAttributeResponse.Builder responseBuilder = UserAttributeResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Update a user attribute definition. + * + */ + @Override + public void updateUserAttribute(UpdateUserAttributeReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.patch("/user_attributes/{user_attribute_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + UpdateUserAttributeResponse.Builder responseBuilder = UpdateUserAttributeResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Delete a user attribute (admin only). + * + */ + @Override + public void deleteUserAttribute(DeleteUserAttributeReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.delete("/user_attributes/{user_attribute_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + DeleteUserAttributeResponse.Builder responseBuilder = DeleteUserAttributeResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Returns all values of a user attribute defined by user groups, in precedence order. + * + * A user may be a member of multiple groups which define different values for a given user attribute. + * The order of group-values in the response determines precedence for selecting which group-value applies + * to a given user. For more information, see [Set User Attribute Group Values](#!/UserAttribute/set_user_attribute_group_values). + * + * Results will only include groups that the caller's user account has permission to see. + * + */ + @Override + public void allUserAttributeGroupValues(AllUserAttributeGroupValuesReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/user_attributes/{user_attribute_id}/group_values", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllUserAttributeGroupValuesResponse.Builder responseBuilder = AllUserAttributeGroupValuesResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + AllUserAttributeGroupValuesStreamResponse.Builder responseBuilder2 = AllUserAttributeGroupValuesStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Define values for a user attribute across a set of groups, in priority order. + * + * This function defines all values for a user attribute defined by user groups. This is a global setting, potentially affecting + * all users in the system. This function replaces any existing group value definitions for the indicated user attribute. + * + * The value of a user attribute for a given user is determined by searching the following locations, in this order: + * + * 1. the user's account settings + * 2. the groups that the user is a member of + * 3. the default value of the user attribute, if any + * + * The user may be a member of multiple groups which define different values for that user attribute. The order of items in the group_values parameter + * determines which group takes priority for that user. Lowest array index wins. + * + * An alternate method to indicate the selection precedence of group-values is to assign numbers to the 'rank' property of each + * group-value object in the array. Lowest 'rank' value wins. If you use this technique, you must assign a + * rank value to every group-value object in the array. + * + * To set a user attribute value for a single user, see [Set User Attribute User Value](#!/User/set_user_attribute_user_value). + * To set a user attribute value for all members of a group, see [Set User Attribute Group Value](#!/Group/update_user_attribute_group_value). + * + */ + @Override + public void setUserAttributeGroupValues(SetUserAttributeGroupValuesReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.post("/user_attributes/{user_attribute_id}/group_values", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + SetUserAttributeGroupValuesResponse.Builder responseBuilder = SetUserAttributeGroupValuesResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + SetUserAttributeGroupValuesStreamResponse.Builder responseBuilder2 = SetUserAttributeGroupValuesStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + //#endregion UserAttribute: Manage User Attributes + + //#region Workspace: Manage Workspaces + + /** + * ### Get All Workspaces + * + * Returns all workspaces available to the calling user. + * + */ + @Override + public void allWorkspaces(AllWorkspacesReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/workspaces", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + AllWorkspacesResponse.Builder responseBuilder = AllWorkspacesResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseBuilder.getResultList().forEach(entry -> { + AllWorkspacesStreamResponse.Builder responseBuilder2 = AllWorkspacesStreamResponse.newBuilder(); + responseBuilder2.setResult(entry); + responseObserver.onNext(responseBuilder2.build()); + }); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + /** + * ### Get A Workspace + * + * Returns information about a workspace such as the git status and selected branches + * of all projects available to the caller's user account. + * + * A workspace defines which versions of project files will be used to evaluate expressions + * and operations that use model definitions - operations such as running queries or rendering dashboards. + * Each project has its own git repository, and each project in a workspace may be configured to reference + * particular branch or revision within their respective repositories. + * + * There are two predefined workspaces available: "production" and "dev". + * + * The production workspace is shared across all Looker users. Models in the production workspace are read-only. + * Changing files in production is accomplished by modifying files in a git branch and using Pull Requests + * to merge the changes from the dev branch into the production branch, and then telling + * Looker to sync with production. + * + * The dev workspace is local to each Looker user. Changes made to project/model files in the dev workspace only affect + * that user, and only when the dev workspace is selected as the active workspace for the API session. + * (See set_session_workspace()). + * + * The dev workspace is NOT unique to an API session. Two applications accessing the Looker API using + * the same user account will see the same files in the dev workspace. To avoid collisions between + * API clients it's best to have each client login with API3 credentials for a different user account. + * + * Changes made to files in a dev workspace are persistent across API sessions. It's a good + * idea to commit any changes you've made to the git repository, but not strictly required. Your modified files + * reside in a special user-specific directory on the Looker server and will still be there when you login in again + * later and use update_session(workspace_id: "dev") to select the dev workspace for the new API session. + * + */ + @Override + public void workspace(WorkspaceReq request, StreamObserver responseObserver) { + try { + String inputJson = JsonFormat + .printer() + .preservingProtoFieldNames() + .print(request); + LookerClientResponse lookerResponse = lookerClient.get("/workspaces/{workspace_id}", inputJson); + Status lookerStatus = lookerResponse.getStatus(); + if (lookerStatus != null) { + responseObserver.onError(lookerStatus.asRuntimeException()); + } else { + WorkspaceResponse.Builder responseBuilder = WorkspaceResponse.newBuilder(); + String outputJson = lookerResponse.getJsonResponse(); + if (outputJson != null) { + JsonFormat + .parser() + .ignoringUnknownFields() + .merge(outputJson, responseBuilder); + } + responseObserver.onNext(responseBuilder.build()); + responseObserver.onCompleted(); + } + } catch (InvalidProtocolBufferException e) { + LOGGER.error("invalid protobuf data", e); + responseObserver.onError(Status.INVALID_ARGUMENT.asRuntimeException()); + } + } + + + //#endregion Workspace: Manage Workspaces +} \ No newline at end of file diff --git a/proto/grpc_proxy/src/main/proto/ping/ping.proto b/proto/grpc_proxy/src/main/proto/ping/ping.proto new file mode 100644 index 000000000..e3f3bfcc5 --- /dev/null +++ b/proto/grpc_proxy/src/main/proto/ping/ping.proto @@ -0,0 +1,12 @@ +syntax = "proto3"; + +package looker; + +option java_package = "com.google.looker.grpc.services"; +option java_multiple_files = true; + +import 'ping/ping_model.proto'; + +service PingService { + rpc Ping(PingRequest) returns (PingResponse) {}; +} diff --git a/proto/grpc_proxy/src/main/proto/ping/ping_model.proto b/proto/grpc_proxy/src/main/proto/ping/ping_model.proto new file mode 100644 index 000000000..f5fdc5b77 --- /dev/null +++ b/proto/grpc_proxy/src/main/proto/ping/ping_model.proto @@ -0,0 +1,12 @@ +syntax = "proto3"; + +package looker; + +option java_package = "com.google.looker.server.rtl"; +option java_multiple_files = true; + +message PingRequest {} + +message PingResponse { + bool active = 1; +} diff --git a/proto/grpc_proxy/src/main/proto/sdk/methods.proto b/proto/grpc_proxy/src/main/proto/sdk/methods.proto new file mode 100644 index 000000000..718438a4e --- /dev/null +++ b/proto/grpc_proxy/src/main/proto/sdk/methods.proto @@ -0,0 +1,3684 @@ +// MIT License +// +// Copyright (c) 2021 Looker Data Sciences, Inc. +// +// Permission is hereby granted, free of charge, to any person obtaining a copy +// of this software and associated documentation files (the "Software"), to deal +// in the Software without restriction, including without limitation the rights +// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +// copies of the Software, and to permit persons to whom the Software is +// furnished to do so, subject to the following conditions: +// +// The above copyright notice and this permission notice shall be included in all +// copies or substantial portions of the Software. +// +// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +// SOFTWARE. +// + +// 438 API methods + + +syntax = "proto3"; + +package looker; + +option java_package = "com.google.looker.grpc.services"; +option java_multiple_files = true; + +import 'sdk/models.proto'; + +service LookerService { + + // Alert: Alert + + // ### Search Alerts + // + rpc SearchAlerts(SearchAlertsReq) returns (SearchAlertsResponse); + + // ### Get an alert by a given alert ID + // + rpc GetAlert(GetAlertReq) returns (GetAlertResponse); + + // ### Update an alert + // # Required fields: `owner_id`, `field`, `destinations`, `comparison_type`, `threshold`, `cron` + // # + // + rpc UpdateAlert(UpdateAlertReq) returns (UpdateAlertResponse); + + // ### Update select alert fields + // # Available fields: `owner_id`, `is_disabled`, `disabled_reason`, `is_public`, `threshold` + // # + // + rpc UpdateAlertField(UpdateAlertFieldReq) returns (UpdateAlertFieldResponse); + + // ### Delete an alert by a given alert ID + // + rpc DeleteAlert(DeleteAlertReq) returns (DeleteAlertResponse); + + // ### Create a new alert and return details of the newly created object + // + // Required fields: `field`, `destinations`, `comparison_type`, `threshold`, `cron` + // + // Example Request: + // Run alert on dashboard element '103' at 5am every day. Send an email to 'test@test.com' if inventory for Los Angeles (using dashboard filter `Warehouse Name`) is lower than 1,000 + // ``` + // { + // "cron": "0 5 * * *", + // "custom_title": "Alert when LA inventory is low", + // "dashboard_element_id": 103, + // "applied_dashboard_filters": [ + // { + // "filter_title": "Warehouse Name", + // "field_name": "distribution_centers.name", + // "filter_value": "Los Angeles CA", + // "filter_description": "is Los Angeles CA" + // } + // ], + // "comparison_type": "LESS_THAN", + // "destinations": [ + // { + // "destination_type": "EMAIL", + // "email_address": "test@test.com" + // } + // ], + // "field": { + // "title": "Number on Hand", + // "name": "inventory_items.number_on_hand" + // }, + // "is_disabled": false, + // "is_public": true, + // "threshold": 1000 + // } + // ``` + // + rpc CreateAlert(CreateAlertReq) returns (CreateAlertResponse); + + // ### Enqueue an Alert by ID + // + rpc EnqueueAlert(EnqueueAlertReq) returns (EnqueueAlertResponse); + + + + // ApiAuth: API Authentication + + // ### Present client credentials to obtain an authorization token + // + // Looker API implements the OAuth2 [Resource Owner Password Credentials Grant](https://looker.com/docs/r/api/outh2_resource_owner_pc) pattern. + // The client credentials required for this login must be obtained by creating an API3 key on a user account + // in the Looker Admin console. The API3 key consists of a public `client_id` and a private `client_secret`. + // + // The access token returned by `login` must be used in the HTTP Authorization header of subsequent + // API requests, like this: + // ``` + // Authorization: token 4QDkCyCtZzYgj4C2p2cj3csJH7zqS5RzKs2kTnG4 + // ``` + // Replace "4QDkCy..." with the `access_token` value returned by `login`. + // The word `token` is a string literal and must be included exactly as shown. + // + // This function can accept `client_id` and `client_secret` parameters as URL query params or as www-form-urlencoded params in the body of the HTTP request. Since there is a small risk that URL parameters may be visible to intermediate nodes on the network route (proxies, routers, etc), passing credentials in the body of the request is considered more secure than URL params. + // + // Example of passing credentials in the HTTP request body: + // ```` + // POST HTTP /login + // Content-Type: application/x-www-form-urlencoded + // + // client_id=CGc9B7v7J48dQSJvxxx&client_secret=nNVS9cSS3xNpSC9JdsBvvvvv + // ```` + // + // ### Best Practice: + // Always pass credentials in body params. Pass credentials in URL query params **only** when you cannot pass body params due to application, tool, or other limitations. + // + // For more information and detailed examples of Looker API authorization, see [How to Authenticate to Looker API3](https://github.com/looker/looker-sdk-ruby/blob/master/authentication.md). + // + rpc Login(LoginReq) returns (LoginResponse); + + // ### Create an access token that runs as a given user. + // + // This can only be called by an authenticated admin user. It allows that admin to generate a new + // authentication token for the user with the given user id. That token can then be used for subsequent + // API calls - which are then performed *as* that target user. + // + // The target user does *not* need to have a pre-existing API client_id/client_secret pair. And, no such + // credentials are created by this call. + // + // This allows for building systems where api user authentication for an arbitrary number of users is done + // outside of Looker and funneled through a single 'service account' with admin permissions. Note that a + // new access token is generated on each call. If target users are going to be making numerous API + // calls in a short period then it is wise to cache this authentication token rather than call this before + // each of those API calls. + // + // See 'login' for more detail on the access token and how to use it. + // + rpc LoginUser(LoginUserReq) returns (LoginUserResponse); + + // ### Logout of the API and invalidate the current access token. + // + rpc Logout(LogoutReq) returns (LogoutResponse); + + + + // Auth: Manage User Authentication Configuration + + // ### Create an embed secret using the specified information. + // + // The value of the `secret` field will be set by Looker and returned. + // + rpc CreateEmbedSecret(CreateEmbedSecretReq) returns (CreateEmbedSecretResponse); + + // ### Delete an embed secret. + // + rpc DeleteEmbedSecret(DeleteEmbedSecretReq) returns (DeleteEmbedSecretResponse); + + // ### Create SSO Embed URL + // + // Creates an SSO embed URL and cryptographically signs it with an embed secret. + // This signed URL can then be used to instantiate a Looker embed session in a PBL web application. + // Do not make any modifications to this URL - any change may invalidate the signature and + // cause the URL to fail to load a Looker embed session. + // + // A signed SSO embed URL can only be used once. After it has been used to request a page from the + // Looker server, the URL is invalid. Future requests using the same URL will fail. This is to prevent + // 'replay attacks'. + // + // The `target_url` property must be a complete URL of a Looker UI page - scheme, hostname, path and query params. + // To load a dashboard with id 56 and with a filter of `Date=1 years`, the looker URL would look like `https:/myname.looker.com/dashboards/56?Date=1%20years`. + // The best way to obtain this target_url is to navigate to the desired Looker page in your web browser, + // copy the URL shown in the browser address bar and paste it into the `target_url` property as a quoted string value in this API request. + // + // Permissions for the embed user are defined by the groups in which the embed user is a member (group_ids property) + // and the lists of models and permissions assigned to the embed user. + // At a minimum, you must provide values for either the group_ids property, or both the models and permissions properties. + // These properties are additive; an embed user can be a member of certain groups AND be granted access to models and permissions. + // + // The embed user's access is the union of permissions granted by the group_ids, models, and permissions properties. + // + // This function does not strictly require all group_ids, user attribute names, or model names to exist at the moment the + // SSO embed url is created. Unknown group_id, user attribute names or model names will be passed through to the output URL. + // To diagnose potential problems with an SSO embed URL, you can copy the signed URL into the Embed URI Validator text box in `/admin/embed`. + // + // The `secret_id` parameter is optional. If specified, its value must be the id of an active secret defined in the Looker instance. + // if not specified, the URL will be signed using the newest active secret defined in the Looker instance. + // + // #### Security Note + // Protect this signed URL as you would an access token or password credentials - do not write + // it to disk, do not pass it to a third party, and only pass it through a secure HTTPS + // encrypted transport. + // + rpc CreateSsoEmbedUrl(CreateSsoEmbedUrlReq) returns (CreateSsoEmbedUrlResponse); + + // ### Create an Embed URL + // + // Creates an embed URL that runs as the Looker user making this API call. ("Embed as me") + // This embed URL can then be used to instantiate a Looker embed session in a + // "Powered by Looker" (PBL) web application. + // + // This is similar to Private Embedding (https://docs.looker.com/r/admin/embed/private-embed). Instead of + // of logging into the Web UI to authenticate, the user has already authenticated against the API to be able to + // make this call. However, unlike Private Embed where the user has access to any other part of the Looker UI, + // the embed web session created by requesting the EmbedUrlResponse.url in a browser only has access to + // content visible under the `/embed` context. + // + // An embed URL can only be used once, and must be used within 5 minutes of being created. After it + // has been used to request a page from the Looker server, the URL is invalid. Future requests using + // the same URL will fail. This is to prevent 'replay attacks'. + // + // The `target_url` property must be a complete URL of a Looker Embedded UI page - scheme, hostname, path starting with "/embed" and query params. + // To load a dashboard with id 56 and with a filter of `Date=1 years`, the looker Embed URL would look like `https://myname.looker.com/embed/dashboards/56?Date=1%20years`. + // The best way to obtain this target_url is to navigate to the desired Looker page in your web browser, + // copy the URL shown in the browser address bar, insert "/embed" after the host/port, and paste it into the `target_url` property as a quoted string value in this API request. + // + // #### Security Note + // Protect this embed URL as you would an access token or password credentials - do not write + // it to disk, do not pass it to a third party, and only pass it through a secure HTTPS + // encrypted transport. + // + rpc CreateEmbedUrlAsMe(CreateEmbedUrlAsMeReq) returns (CreateEmbedUrlAsMeResponse); + + // ### Get the LDAP configuration. + // + // Looker can be optionally configured to authenticate users against an Active Directory or other LDAP directory server. + // LDAP setup requires coordination with an administrator of that directory server. + // + // Only Looker administrators can read and update the LDAP configuration. + // + // Configuring LDAP impacts authentication for all users. This configuration should be done carefully. + // + // Looker maintains a single LDAP configuration. It can be read and updated. Updates only succeed if the new state will be valid (in the sense that all required fields are populated); it is up to you to ensure that the configuration is appropriate and correct). + // + // LDAP is enabled or disabled for Looker using the **enabled** field. + // + // Looker will never return an **auth_password** field. That value can be set, but never retrieved. + // + // See the [Looker LDAP docs](https://www.looker.com/docs/r/api/ldap_setup) for additional information. + // + rpc LdapConfig(LdapConfigReq) returns (LdapConfigResponse); + + // ### Update the LDAP configuration. + // + // Configuring LDAP impacts authentication for all users. This configuration should be done carefully. + // + // Only Looker administrators can read and update the LDAP configuration. + // + // LDAP is enabled or disabled for Looker using the **enabled** field. + // + // It is **highly** recommended that any LDAP setting changes be tested using the APIs below before being set globally. + // + // See the [Looker LDAP docs](https://www.looker.com/docs/r/api/ldap_setup) for additional information. + // + rpc UpdateLdapConfig(UpdateLdapConfigReq) returns (UpdateLdapConfigResponse); + + // ### Test the connection settings for an LDAP configuration. + // + // This tests that the connection is possible given a connection_host and connection_port. + // + // **connection_host** and **connection_port** are required. **connection_tls** is optional. + // + // Example: + // ```json + // { + // "connection_host": "ldap.example.com", + // "connection_port": "636", + // "connection_tls": true + // } + // ``` + // + // No authentication to the LDAP server is attempted. + // + // The active LDAP settings are not modified. + // + rpc TestLdapConfigConnection(TestLdapConfigConnectionReq) returns (TestLdapConfigConnectionResponse); + + // ### Test the connection authentication settings for an LDAP configuration. + // + // This tests that the connection is possible and that a 'server' account to be used by Looker can authenticate to the LDAP server given connection and authentication information. + // + // **connection_host**, **connection_port**, and **auth_username**, are required. **connection_tls** and **auth_password** are optional. + // + // Example: + // ```json + // { + // "connection_host": "ldap.example.com", + // "connection_port": "636", + // "connection_tls": true, + // "auth_username": "cn=looker,dc=example,dc=com", + // "auth_password": "secret" + // } + // ``` + // + // Looker will never return an **auth_password**. If this request omits the **auth_password** field, then the **auth_password** value from the active config (if present) will be used for the test. + // + // The active LDAP settings are not modified. + // + // + rpc TestLdapConfigAuth(TestLdapConfigAuthReq) returns (TestLdapConfigAuthResponse); + + // ### Test the user authentication settings for an LDAP configuration without authenticating the user. + // + // This test will let you easily test the mapping for user properties and roles for any user without needing to authenticate as that user. + // + // This test accepts a full LDAP configuration along with a username and attempts to find the full info for the user from the LDAP server without actually authenticating the user. So, user password is not required.The configuration is validated before attempting to contact the server. + // + // **test_ldap_user** is required. + // + // The active LDAP settings are not modified. + // + // + rpc TestLdapConfigUserInfo(TestLdapConfigUserInfoReq) returns (TestLdapConfigUserInfoResponse); + + // ### Test the user authentication settings for an LDAP configuration. + // + // This test accepts a full LDAP configuration along with a username/password pair and attempts to authenticate the user with the LDAP server. The configuration is validated before attempting the authentication. + // + // Looker will never return an **auth_password**. If this request omits the **auth_password** field, then the **auth_password** value from the active config (if present) will be used for the test. + // + // **test_ldap_user** and **test_ldap_password** are required. + // + // The active LDAP settings are not modified. + // + // + rpc TestLdapConfigUserAuth(TestLdapConfigUserAuthReq) returns (TestLdapConfigUserAuthResponse); + + // ### List All OAuth Client Apps + // + // Lists all applications registered to use OAuth2 login with this Looker instance, including + // enabled and disabled apps. + // + // Results are filtered to include only the apps that the caller (current user) + // has permission to see. + // + rpc AllOauthClientApps(AllOauthClientAppsReq) returns (AllOauthClientAppsResponse); + + // ### Get Oauth Client App + // + // Returns the registered app client with matching client_guid. + // + rpc OauthClientApp(OauthClientAppReq) returns (OauthClientAppResponse); + + // ### Register an OAuth2 Client App + // + // Registers details identifying an external web app or native app as an OAuth2 login client of the Looker instance. + // The app registration must provide a unique client_guid and redirect_uri that the app will present + // in OAuth login requests. If the client_guid and redirect_uri parameters in the login request do not match + // the app details registered with the Looker instance, the request is assumed to be a forgery and is rejected. + // + rpc RegisterOauthClientApp(RegisterOauthClientAppReq) returns (RegisterOauthClientAppResponse); + + // ### Update OAuth2 Client App Details + // + // Modifies the details a previously registered OAuth2 login client app. + // + rpc UpdateOauthClientApp(UpdateOauthClientAppReq) returns (UpdateOauthClientAppResponse); + + // ### Delete OAuth Client App + // + // Deletes the registration info of the app with the matching client_guid. + // All active sessions and tokens issued for this app will immediately become invalid. + // + // ### Note: this deletion cannot be undone. + // + rpc DeleteOauthClientApp(DeleteOauthClientAppReq) returns (DeleteOauthClientAppResponse); + + // ### Invalidate All Issued Tokens + // + // Immediately invalidates all auth codes, sessions, access tokens and refresh tokens issued for + // this app for ALL USERS of this app. + // + rpc InvalidateTokens(InvalidateTokensReq) returns (InvalidateTokensResponse); + + // ### Activate an app for a user + // + // Activates a user for a given oauth client app. This indicates the user has been informed that + // the app will have access to the user's looker data, and that the user has accepted and allowed + // the app to use their Looker account. + // + // Activating a user for an app that the user is already activated with returns a success response. + // + rpc ActivateAppUser(ActivateAppUserReq) returns (ActivateAppUserResponse); + + // ### Deactivate an app for a user + // + // Deactivate a user for a given oauth client app. All tokens issued to the app for + // this user will be invalid immediately. Before the user can use the app with their + // Looker account, the user will have to read and accept an account use disclosure statement for the app. + // + // Admin users can deactivate other users, but non-admin users can only deactivate themselves. + // + // As with most REST DELETE operations, this endpoint does not return an error if the indicated + // resource (app or user) does not exist or has already been deactivated. + // + rpc DeactivateAppUser(DeactivateAppUserReq) returns (DeactivateAppUserResponse); + + // ### Get the OIDC configuration. + // + // Looker can be optionally configured to authenticate users against an OpenID Connect (OIDC) + // authentication server. OIDC setup requires coordination with an administrator of that server. + // + // Only Looker administrators can read and update the OIDC configuration. + // + // Configuring OIDC impacts authentication for all users. This configuration should be done carefully. + // + // Looker maintains a single OIDC configuation. It can be read and updated. Updates only succeed if the new state will be valid (in the sense that all required fields are populated); it is up to you to ensure that the configuration is appropriate and correct). + // + // OIDC is enabled or disabled for Looker using the **enabled** field. + // + rpc OidcConfig(OidcConfigReq) returns (OidcConfigResponse); + + // ### Update the OIDC configuration. + // + // Configuring OIDC impacts authentication for all users. This configuration should be done carefully. + // + // Only Looker administrators can read and update the OIDC configuration. + // + // OIDC is enabled or disabled for Looker using the **enabled** field. + // + // It is **highly** recommended that any OIDC setting changes be tested using the APIs below before being set globally. + // + rpc UpdateOidcConfig(UpdateOidcConfigReq) returns (UpdateOidcConfigResponse); + + // ### Get a OIDC test configuration by test_slug. + // + rpc OidcTestConfig(OidcTestConfigReq) returns (OidcTestConfigResponse); + + // ### Delete a OIDC test configuration. + // + rpc DeleteOidcTestConfig(DeleteOidcTestConfigReq) returns (DeleteOidcTestConfigResponse); + + // ### Create a OIDC test configuration. + // + rpc CreateOidcTestConfig(CreateOidcTestConfigReq) returns (CreateOidcTestConfigResponse); + + // ### Get password config. + // + rpc PasswordConfig(PasswordConfigReq) returns (PasswordConfigResponse); + + // ### Update password config. + // + rpc UpdatePasswordConfig(UpdatePasswordConfigReq) returns (UpdatePasswordConfigResponse); + + // ### Force all credentials_email users to reset their login passwords upon their next login. + // + rpc ForcePasswordResetAtNextLoginForAllUsers(ForcePasswordResetAtNextLoginForAllUsersReq) returns (ForcePasswordResetAtNextLoginForAllUsersResponse); + + // ### Get the SAML configuration. + // + // Looker can be optionally configured to authenticate users against a SAML authentication server. + // SAML setup requires coordination with an administrator of that server. + // + // Only Looker administrators can read and update the SAML configuration. + // + // Configuring SAML impacts authentication for all users. This configuration should be done carefully. + // + // Looker maintains a single SAML configuation. It can be read and updated. Updates only succeed if the new state will be valid (in the sense that all required fields are populated); it is up to you to ensure that the configuration is appropriate and correct). + // + // SAML is enabled or disabled for Looker using the **enabled** field. + // + rpc SamlConfig(SamlConfigReq) returns (SamlConfigResponse); + + // ### Update the SAML configuration. + // + // Configuring SAML impacts authentication for all users. This configuration should be done carefully. + // + // Only Looker administrators can read and update the SAML configuration. + // + // SAML is enabled or disabled for Looker using the **enabled** field. + // + // It is **highly** recommended that any SAML setting changes be tested using the APIs below before being set globally. + // + rpc UpdateSamlConfig(UpdateSamlConfigReq) returns (UpdateSamlConfigResponse); + + // ### Get a SAML test configuration by test_slug. + // + rpc SamlTestConfig(SamlTestConfigReq) returns (SamlTestConfigResponse); + + // ### Delete a SAML test configuration. + // + rpc DeleteSamlTestConfig(DeleteSamlTestConfigReq) returns (DeleteSamlTestConfigResponse); + + // ### Create a SAML test configuration. + // + rpc CreateSamlTestConfig(CreateSamlTestConfigReq) returns (CreateSamlTestConfigResponse); + + // ### Parse the given xml as a SAML IdP metadata document and return the result. + // + rpc ParseSamlIdpMetadata(ParseSamlIdpMetadataReq) returns (ParseSamlIdpMetadataResponse); + + // ### Fetch the given url and parse it as a SAML IdP metadata document and return the result. + // Note that this requires that the url be public or at least at a location where the Looker instance + // can fetch it without requiring any special authentication. + // + rpc FetchAndParseSamlIdpMetadata(FetchAndParseSamlIdpMetadataReq) returns (FetchAndParseSamlIdpMetadataResponse); + + // ### Get session config. + // + rpc SessionConfig(SessionConfigReq) returns (SessionConfigResponse); + + // ### Update session config. + // + rpc UpdateSessionConfig(UpdateSessionConfigReq) returns (UpdateSessionConfigResponse); + + // ### Get Support Access Allowlist Users + // + // Returns the users that have been added to the Support Access Allowlist + // + rpc GetSupportAccessAllowlistEntries(GetSupportAccessAllowlistEntriesReq) returns (GetSupportAccessAllowlistEntriesResponse); + + // ### Add Support Access Allowlist Users + // + // Adds a list of emails to the Allowlist, using the provided reason + // + rpc AddSupportAccessAllowlistEntries(AddSupportAccessAllowlistEntriesReq) returns (AddSupportAccessAllowlistEntriesResponse); + + // ### Delete Support Access Allowlist User + // + // Deletes the specified Allowlist Entry Id + // + rpc DeleteSupportAccessAllowlistEntry(DeleteSupportAccessAllowlistEntryReq) returns (DeleteSupportAccessAllowlistEntryResponse); + + // ### Enable Support Access + // + // Enables Support Access for the provided duration + // + rpc EnableSupportAccess(EnableSupportAccessReq) returns (EnableSupportAccessResponse); + + // ### Disable Support Access + // + // Disables Support Access immediately + // + rpc DisableSupportAccess(DisableSupportAccessReq) returns (DisableSupportAccessResponse); + + // ### Support Access Status + // + // Returns the current Support Access Status + // + rpc SupportAccessStatus(SupportAccessStatusReq) returns (SupportAccessStatusResponse); + + // ### Get currently locked-out users. + // + rpc AllUserLoginLockouts(AllUserLoginLockoutsReq) returns (AllUserLoginLockoutsResponse); + + // ### Search currently locked-out users. + // + rpc SearchUserLoginLockouts(SearchUserLoginLockoutsReq) returns (SearchUserLoginLockoutsResponse); + + // ### Removes login lockout for the associated user. + // + rpc DeleteUserLoginLockout(DeleteUserLoginLockoutReq) returns (DeleteUserLoginLockoutResponse); + + + + // Board: Manage Boards + + // ### Get information about all boards. + // + rpc AllBoards(AllBoardsReq) returns (AllBoardsResponse); + + // ### Create a new board. + // + rpc CreateBoard(CreateBoardReq) returns (CreateBoardResponse); + + // ### Search Boards + // + // If multiple search params are given and `filter_or` is FALSE or not specified, + // search params are combined in a logical AND operation. + // Only rows that match *all* search param criteria will be returned. + // + // If `filter_or` is TRUE, multiple search params are combined in a logical OR operation. + // Results will include rows that match **any** of the search criteria. + // + // String search params use case-insensitive matching. + // String search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions. + // example="dan%" will match "danger" and "Danzig" but not "David" + // example="D_m%" will match "Damage" and "dump" + // + // Integer search params can accept a single value or a comma separated list of values. The multiple + // values will be combined under a logical OR operation - results will match at least one of + // the given values. + // + // Most search params can accept "IS NULL" and "NOT NULL" as special expressions to match + // or exclude (respectively) rows where the column is null. + // + // Boolean search params accept only "true" and "false" as values. + // + // + rpc SearchBoards(SearchBoardsReq) returns (SearchBoardsResponse); + + // ### Get information about a board. + // + rpc Board(BoardReq) returns (BoardResponse); + + // ### Update a board definition. + // + rpc UpdateBoard(UpdateBoardReq) returns (UpdateBoardResponse); + + // ### Delete a board. + // + rpc DeleteBoard(DeleteBoardReq) returns (DeleteBoardResponse); + + // ### Get information about all board items. + // + rpc AllBoardItems(AllBoardItemsReq) returns (AllBoardItemsResponse); + + // ### Create a new board item. + // + rpc CreateBoardItem(CreateBoardItemReq) returns (CreateBoardItemResponse); + + // ### Get information about a board item. + // + rpc BoardItem(BoardItemReq) returns (BoardItemResponse); + + // ### Update a board item definition. + // + rpc UpdateBoardItem(UpdateBoardItemReq) returns (UpdateBoardItemResponse); + + // ### Delete a board item. + // + rpc DeleteBoardItem(DeleteBoardItemReq) returns (DeleteBoardItemResponse); + + // ### Get information about all board sections. + // + rpc AllBoardSections(AllBoardSectionsReq) returns (AllBoardSectionsResponse); + + // ### Create a new board section. + // + rpc CreateBoardSection(CreateBoardSectionReq) returns (CreateBoardSectionResponse); + + // ### Get information about a board section. + // + rpc BoardSection(BoardSectionReq) returns (BoardSectionResponse); + + // ### Update a board section definition. + // + rpc UpdateBoardSection(UpdateBoardSectionReq) returns (UpdateBoardSectionResponse); + + // ### Delete a board section. + // + rpc DeleteBoardSection(DeleteBoardSectionReq) returns (DeleteBoardSectionResponse); + + + + // ColorCollection: Manage Color Collections + + // ### Get an array of all existing Color Collections + // Get a **single** color collection by id with [ColorCollection](#!/ColorCollection/color_collection) + // + // Get all **standard** color collections with [ColorCollection](#!/ColorCollection/color_collections_standard) + // + // Get all **custom** color collections with [ColorCollection](#!/ColorCollection/color_collections_custom) + // + // **Note**: Only an API user with the Admin role can call this endpoint. Unauthorized requests will return `Not Found` (404) errors. + // + // + rpc AllColorCollections(AllColorCollectionsReq) returns (AllColorCollectionsResponse); + + // ### Create a custom color collection with the specified information + // + // Creates a new custom color collection object, returning the details, including the created id. + // + // **Update** an existing color collection with [Update Color Collection](#!/ColorCollection/update_color_collection) + // + // **Permanently delete** an existing custom color collection with [Delete Color Collection](#!/ColorCollection/delete_color_collection) + // + // **Note**: Only an API user with the Admin role can call this endpoint. Unauthorized requests will return `Not Found` (404) errors. + // + // + rpc CreateColorCollection(CreateColorCollectionReq) returns (CreateColorCollectionResponse); + + // ### Get an array of all existing **Custom** Color Collections + // Get a **single** color collection by id with [ColorCollection](#!/ColorCollection/color_collection) + // + // Get all **standard** color collections with [ColorCollection](#!/ColorCollection/color_collections_standard) + // + // **Note**: Only an API user with the Admin role can call this endpoint. Unauthorized requests will return `Not Found` (404) errors. + // + // + rpc ColorCollectionsCustom(ColorCollectionsCustomReq) returns (ColorCollectionsCustomResponse); + + // ### Get an array of all existing **Standard** Color Collections + // Get a **single** color collection by id with [ColorCollection](#!/ColorCollection/color_collection) + // + // Get all **custom** color collections with [ColorCollection](#!/ColorCollection/color_collections_custom) + // + // **Note**: Only an API user with the Admin role can call this endpoint. Unauthorized requests will return `Not Found` (404) errors. + // + // + rpc ColorCollectionsStandard(ColorCollectionsStandardReq) returns (ColorCollectionsStandardResponse); + + // ### Get the default color collection + // + // Use this to retrieve the default Color Collection. + // + // Set the default color collection with [ColorCollection](#!/ColorCollection/set_default_color_collection) + // + rpc DefaultColorCollection(DefaultColorCollectionReq) returns (DefaultColorCollectionResponse); + + // ### Set the global default Color Collection by ID + // + // Returns the new specified default Color Collection object. + // **Note**: Only an API user with the Admin role can call this endpoint. Unauthorized requests will return `Not Found` (404) errors. + // + // + rpc SetDefaultColorCollection(SetDefaultColorCollectionReq) returns (SetDefaultColorCollectionResponse); + + // ### Get a Color Collection by ID + // + // Use this to retrieve a specific Color Collection. + // Get a **single** color collection by id with [ColorCollection](#!/ColorCollection/color_collection) + // + // Get all **standard** color collections with [ColorCollection](#!/ColorCollection/color_collections_standard) + // + // Get all **custom** color collections with [ColorCollection](#!/ColorCollection/color_collections_custom) + // + // **Note**: Only an API user with the Admin role can call this endpoint. Unauthorized requests will return `Not Found` (404) errors. + // + // + rpc ColorCollection(ColorCollectionReq) returns (ColorCollectionResponse); + + // ### Update a custom color collection by id. + // **Note**: Only an API user with the Admin role can call this endpoint. Unauthorized requests will return `Not Found` (404) errors. + // + // + rpc UpdateColorCollection(UpdateColorCollectionReq) returns (UpdateColorCollectionResponse); + + // ### Delete a custom color collection by id + // + // This operation permanently deletes the identified **Custom** color collection. + // + // **Standard** color collections cannot be deleted + // + // Because multiple color collections can have the same label, they must be deleted by ID, not name. + // **Note**: Only an API user with the Admin role can call this endpoint. Unauthorized requests will return `Not Found` (404) errors. + // + // + rpc DeleteColorCollection(DeleteColorCollectionReq) returns (DeleteColorCollectionResponse); + + + + // Config: Manage General Configuration + + // Get the current Cloud Storage Configuration. + // + rpc CloudStorageConfiguration(CloudStorageConfigurationReq) returns (CloudStorageConfigurationResponse); + + // Update the current Cloud Storage Configuration. + // + rpc UpdateCloudStorageConfiguration(UpdateCloudStorageConfigurationReq) returns (UpdateCloudStorageConfigurationResponse); + + // ### Get the current status and content of custom welcome emails + // + rpc CustomWelcomeEmail(CustomWelcomeEmailReq) returns (CustomWelcomeEmailResponse); + + // Update custom welcome email setting and values. Optionally send a test email with the new content to the currently logged in user. + // + rpc UpdateCustomWelcomeEmail(UpdateCustomWelcomeEmailReq) returns (UpdateCustomWelcomeEmailResponse); + + // Requests to this endpoint will send a welcome email with the custom content provided in the body to the currently logged in user. + // + rpc UpdateCustomWelcomeEmailTest(UpdateCustomWelcomeEmailTestReq) returns (UpdateCustomWelcomeEmailTestResponse); + + // ### Retrieve the value for whether or not digest emails is enabled + // + rpc DigestEmailsEnabled(DigestEmailsEnabledReq) returns (DigestEmailsEnabledResponse); + + // ### Update the setting for enabling/disabling digest emails + // + rpc UpdateDigestEmailsEnabled(UpdateDigestEmailsEnabledReq) returns (UpdateDigestEmailsEnabledResponse); + + // ### Trigger the generation of digest email records and send them to Looker's internal system. This does not send + // any actual emails, it generates records containing content which may be of interest for users who have become inactive. + // Emails will be sent at a later time from Looker's internal system if the Digest Emails feature is enabled in settings. + rpc CreateDigestEmailSend(CreateDigestEmailSendReq) returns (CreateDigestEmailSendResponse); + + // ### Get Egress IP Addresses + // + // Returns the list of public egress IP Addresses for a hosted customer's instance + // + rpc PublicEgressIpAddresses(PublicEgressIpAddressesReq) returns (PublicEgressIpAddressesResponse); + + // ### Set the menu item name and content for internal help resources + // + rpc InternalHelpResourcesContent(InternalHelpResourcesContentReq) returns (InternalHelpResourcesContentResponse); + + // Update internal help resources content + // + rpc UpdateInternalHelpResourcesContent(UpdateInternalHelpResourcesContentReq) returns (UpdateInternalHelpResourcesContentResponse); + + // ### Get and set the options for internal help resources + // + rpc InternalHelpResources(InternalHelpResourcesReq) returns (InternalHelpResourcesResponse); + + // Update internal help resources settings + // + rpc UpdateInternalHelpResources(UpdateInternalHelpResourcesReq) returns (UpdateInternalHelpResourcesResponse); + + // ### Get all legacy features. + // + rpc AllLegacyFeatures(AllLegacyFeaturesReq) returns (AllLegacyFeaturesResponse); + + // ### Get information about the legacy feature with a specific id. + // + rpc LegacyFeature(LegacyFeatureReq) returns (LegacyFeatureResponse); + + // ### Update information about the legacy feature with a specific id. + // + rpc UpdateLegacyFeature(UpdateLegacyFeatureReq) returns (UpdateLegacyFeatureResponse); + + // ### Get a list of locales that Looker supports. + // + rpc AllLocales(AllLocalesReq) returns (AllLocalesResponse); + + // ### Get all mobile settings. + // + rpc MobileSettings(MobileSettingsReq) returns (MobileSettingsResponse); + + // ### Get Looker Settings + // + // Available settings are: + // - extension_framework_enabled + // - marketplace_auto_install_enabled + // - marketplace_enabled + // - privatelabel_configuration + // - custom_welcome_email + // - onboarding_enabled + // + // + rpc GetSetting(GetSettingReq) returns (GetSettingResponse); + + // ### Configure Looker Settings + // + // Available settings are: + // - extension_framework_enabled + // - marketplace_auto_install_enabled + // - marketplace_enabled + // - privatelabel_configuration + // - custom_welcome_email + // - onboarding_enabled + // + // See the `Setting` type for more information on the specific values that can be configured. + // + rpc SetSetting(SetSettingReq) returns (SetSettingResponse); + + // ### Configure SMTP Settings + // This API allows users to configure the SMTP settings on the Looker instance. + // This API is only supported in the OEM jar. Additionally, only admin users are authorised to call this API. + // + rpc SetSmtpSettings(SetSmtpSettingsReq) returns (SetSmtpSettingsResponse); + + // ### Get current SMTP status. + // + rpc SmtpStatus(SmtpStatusReq) returns (SmtpStatusResponse); + + // ### Get a list of timezones that Looker supports (e.g. useful for scheduling tasks). + // + rpc AllTimezones(AllTimezonesReq) returns (AllTimezonesResponse); + + // ### Get information about all API versions supported by this Looker instance. + // + rpc Versions(VersionsReq) returns (VersionsResponse); + + // ### Get an API specification for this Looker instance. + // + // The specification is returned as a JSON document in Swagger 2.x format + // + rpc ApiSpec(ApiSpecReq) returns (ApiSpecResponse); + + // ### This feature is enabled only by special license. + // ### Gets the whitelabel configuration, which includes hiding documentation links, custom favicon uploading, etc. + // + rpc WhitelabelConfiguration(WhitelabelConfigurationReq) returns (WhitelabelConfigurationResponse); + + // ### Update the whitelabel configuration + // + rpc UpdateWhitelabelConfiguration(UpdateWhitelabelConfigurationReq) returns (UpdateWhitelabelConfigurationResponse); + + + + // Connection: Manage Database Connections + + // ### Get information about all connections. + // + rpc AllConnections(AllConnectionsReq) returns (AllConnectionsResponse); + + // ### Create a connection using the specified configuration. + // + rpc CreateConnection(CreateConnectionReq) returns (CreateConnectionResponse); + + // ### Get information about a connection. + // + rpc Connection(ConnectionReq) returns (ConnectionResponse); + + // ### Update a connection using the specified configuration. + // + rpc UpdateConnection(UpdateConnectionReq) returns (UpdateConnectionResponse); + + // ### Delete a connection. + // + rpc DeleteConnection(DeleteConnectionReq) returns (DeleteConnectionResponse); + + // ### Delete a connection override. + // + rpc DeleteConnectionOverride(DeleteConnectionOverrideReq) returns (DeleteConnectionOverrideResponse); + + // ### Test an existing connection. + // + // Note that a connection's 'dialect' property has a 'connection_tests' property that lists the + // specific types of tests that the connection supports. + // + // This API is rate limited. + // + // Unsupported tests in the request will be ignored. + // + rpc TestConnection(TestConnectionReq) returns (TestConnectionResponse); + + // ### Test a connection configuration. + // + // Note that a connection's 'dialect' property has a 'connection_tests' property that lists the + // specific types of tests that the connection supports. + // + // This API is rate limited. + // + // Unsupported tests in the request will be ignored. + // + rpc TestConnectionConfig(TestConnectionConfigReq) returns (TestConnectionConfigResponse); + + // ### Get information about all dialects. + // + rpc AllDialectInfos(AllDialectInfosReq) returns (AllDialectInfosResponse); + + // ### Get all External OAuth Applications. + // + // This is an OAuth Application which Looker uses to access external systems. + // + rpc AllExternalOauthApplications(AllExternalOauthApplicationsReq) returns (AllExternalOauthApplicationsResponse); + + // ### Create an OAuth Application using the specified configuration. + // + // This is an OAuth Application which Looker uses to access external systems. + // + rpc CreateExternalOauthApplication(CreateExternalOauthApplicationReq) returns (CreateExternalOauthApplicationResponse); + + // ### Create OAuth User state. + // + rpc CreateOauthApplicationUserState(CreateOauthApplicationUserStateReq) returns (CreateOauthApplicationUserStateResponse); + + // ### Get information about all SSH Servers. + // + rpc AllSshServers(AllSshServersReq) returns (AllSshServersResponse); + + // ### Create an SSH Server. + // + rpc CreateSshServer(CreateSshServerReq) returns (CreateSshServerResponse); + + // ### Get information about an SSH Server. + // + rpc SshServer(SshServerReq) returns (SshServerResponse); + + // ### Update an SSH Server. + // + rpc UpdateSshServer(UpdateSshServerReq) returns (UpdateSshServerResponse); + + // ### Delete an SSH Server. + // + rpc DeleteSshServer(DeleteSshServerReq) returns (DeleteSshServerResponse); + + // ### Test the SSH Server + // + rpc TestSshServer(TestSshServerReq) returns (TestSshServerResponse); + + // ### Get information about all SSH Tunnels. + // + rpc AllSshTunnels(AllSshTunnelsReq) returns (AllSshTunnelsResponse); + + // ### Create an SSH Tunnel + // + rpc CreateSshTunnel(CreateSshTunnelReq) returns (CreateSshTunnelResponse); + + // ### Get information about an SSH Tunnel. + // + rpc SshTunnel(SshTunnelReq) returns (SshTunnelResponse); + + // ### Update an SSH Tunnel + // + rpc UpdateSshTunnel(UpdateSshTunnelReq) returns (UpdateSshTunnelResponse); + + // ### Delete an SSH Tunnel + // + rpc DeleteSshTunnel(DeleteSshTunnelReq) returns (DeleteSshTunnelResponse); + + // ### Test the SSH Tunnel + // + rpc TestSshTunnel(TestSshTunnelReq) returns (TestSshTunnelResponse); + + // ### Get the SSH public key + // + // Get the public key created for this instance to identify itself to a remote SSH server. + // + rpc SshPublicKey(SshPublicKeyReq) returns (SshPublicKeyResponse); + + + + // Content: Manage Content + + // ### Search Favorite Content + // + // If multiple search params are given and `filter_or` is FALSE or not specified, + // search params are combined in a logical AND operation. + // Only rows that match *all* search param criteria will be returned. + // + // If `filter_or` is TRUE, multiple search params are combined in a logical OR operation. + // Results will include rows that match **any** of the search criteria. + // + // String search params use case-insensitive matching. + // String search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions. + // example="dan%" will match "danger" and "Danzig" but not "David" + // example="D_m%" will match "Damage" and "dump" + // + // Integer search params can accept a single value or a comma separated list of values. The multiple + // values will be combined under a logical OR operation - results will match at least one of + // the given values. + // + // Most search params can accept "IS NULL" and "NOT NULL" as special expressions to match + // or exclude (respectively) rows where the column is null. + // + // Boolean search params accept only "true" and "false" as values. + // + // + rpc SearchContentFavorites(SearchContentFavoritesReq) returns (SearchContentFavoritesResponse); + + // ### Get favorite content by its id + rpc ContentFavorite(ContentFavoriteReq) returns (ContentFavoriteResponse); + + // ### Delete favorite content + rpc DeleteContentFavorite(DeleteContentFavoriteReq) returns (DeleteContentFavoriteResponse); + + // ### Create favorite content + rpc CreateContentFavorite(CreateContentFavoriteReq) returns (CreateContentFavoriteResponse); + + // ### Get information about all content metadata in a space. + // + rpc AllContentMetadatas(AllContentMetadatasReq) returns (AllContentMetadatasResponse); + + // ### Get information about an individual content metadata record. + // + rpc ContentMetadata(ContentMetadataReq) returns (ContentMetadataResponse); + + // ### Move a piece of content. + // + rpc UpdateContentMetadata(UpdateContentMetadataReq) returns (UpdateContentMetadataResponse); + + // ### All content metadata access records for a content metadata item. + // + rpc AllContentMetadataAccesses(AllContentMetadataAccessesReq) returns (AllContentMetadataAccessesResponse); + + // ### Create content metadata access. + // + rpc CreateContentMetadataAccess(CreateContentMetadataAccessReq) returns (CreateContentMetadataAccessResponse); + + // ### Update type of access for content metadata. + // + rpc UpdateContentMetadataAccess(UpdateContentMetadataAccessReq) returns (UpdateContentMetadataAccessResponse); + + // ### Remove content metadata access. + // + rpc DeleteContentMetadataAccess(DeleteContentMetadataAccessReq) returns (DeleteContentMetadataAccessResponse); + + // ### Get an image representing the contents of a dashboard or look. + // + // The returned thumbnail is an abstract representation of the contents of a dashbord or look and does not + // reflect the actual data displayed in the respective visualizations. + // + rpc ContentThumbnail(ContentThumbnailReq) returns (ContentThumbnailResponse); + + // ### Validate All Content + // + // Performs validation of all looks and dashboards + // Returns a list of errors found as well as metadata about the content validation run. + // + rpc ContentValidation(ContentValidationReq) returns (ContentValidationResponse); + + // ### Search Content Views + // + // If multiple search params are given and `filter_or` is FALSE or not specified, + // search params are combined in a logical AND operation. + // Only rows that match *all* search param criteria will be returned. + // + // If `filter_or` is TRUE, multiple search params are combined in a logical OR operation. + // Results will include rows that match **any** of the search criteria. + // + // String search params use case-insensitive matching. + // String search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions. + // example="dan%" will match "danger" and "Danzig" but not "David" + // example="D_m%" will match "Damage" and "dump" + // + // Integer search params can accept a single value or a comma separated list of values. The multiple + // values will be combined under a logical OR operation - results will match at least one of + // the given values. + // + // Most search params can accept "IS NULL" and "NOT NULL" as special expressions to match + // or exclude (respectively) rows where the column is null. + // + // Boolean search params accept only "true" and "false" as values. + // + // + rpc SearchContentViews(SearchContentViewsReq) returns (SearchContentViewsResponse); + + // ### Get a vector image representing the contents of a dashboard or look. + // + // # DEPRECATED: Use [content_thumbnail()](#!/Content/content_thumbnail) + // + // The returned thumbnail is an abstract representation of the contents of a dashbord or look and does not + // reflect the actual data displayed in the respective visualizations. + // + rpc VectorThumbnail(VectorThumbnailReq) returns (VectorThumbnailResponse); + + + + // Dashboard: Manage Dashboards + + // ### Get information about all active dashboards. + // + // Returns an array of **abbreviated dashboard objects**. Dashboards marked as deleted are excluded from this list. + // + // Get the **full details** of a specific dashboard by id with [dashboard()](#!/Dashboard/dashboard) + // + // Find **deleted dashboards** with [search_dashboards()](#!/Dashboard/search_dashboards) + // + rpc AllDashboards(AllDashboardsReq) returns (AllDashboardsResponse); + + // ### Create a new dashboard + // + // Creates a new dashboard object and returns the details of the newly created dashboard. + // + // `Title` and `space_id` are required fields. + // `Space_id` must contain the id of an existing space. + // A dashboard's `title` must be unique within the space in which it resides. + // + // If you receive a 422 error response when creating a dashboard, be sure to look at the + // response body for information about exactly which fields are missing or contain invalid data. + // + // You can **update** an existing dashboard with [update_dashboard()](#!/Dashboard/update_dashboard) + // + // You can **permanently delete** an existing dashboard with [delete_dashboard()](#!/Dashboard/delete_dashboard) + // + rpc CreateDashboard(CreateDashboardReq) returns (CreateDashboardResponse); + + // ### Search Dashboards + // + // Returns an **array of dashboard objects** that match the specified search criteria. + // + // If multiple search params are given and `filter_or` is FALSE or not specified, + // search params are combined in a logical AND operation. + // Only rows that match *all* search param criteria will be returned. + // + // If `filter_or` is TRUE, multiple search params are combined in a logical OR operation. + // Results will include rows that match **any** of the search criteria. + // + // String search params use case-insensitive matching. + // String search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions. + // example="dan%" will match "danger" and "Danzig" but not "David" + // example="D_m%" will match "Damage" and "dump" + // + // Integer search params can accept a single value or a comma separated list of values. The multiple + // values will be combined under a logical OR operation - results will match at least one of + // the given values. + // + // Most search params can accept "IS NULL" and "NOT NULL" as special expressions to match + // or exclude (respectively) rows where the column is null. + // + // Boolean search params accept only "true" and "false" as values. + // + // + // The parameters `limit`, and `offset` are recommended for fetching results in page-size chunks. + // + // Get a **single dashboard** by id with [dashboard()](#!/Dashboard/dashboard) + // + rpc SearchDashboards(SearchDashboardsReq) returns (SearchDashboardsResponse); + + // ### Import a LookML dashboard to a space as a UDD + // Creates a UDD (a dashboard which exists in the Looker database rather than as a LookML file) from the LookML dashboard + // and places it in the space specified. The created UDD will have a lookml_link_id which links to the original LookML dashboard. + // + // To give the imported dashboard specify a (e.g. title: "my title") in the body of your request, otherwise the imported + // dashboard will have the same title as the original LookML dashboard. + // + // For this operation to succeed the user must have permission to see the LookML dashboard in question, and have permission to + // create content in the space the dashboard is being imported to. + // + // **Sync** a linked UDD with [sync_lookml_dashboard()](#!/Dashboard/sync_lookml_dashboard) + // **Unlink** a linked UDD by setting lookml_link_id to null with [update_dashboard()](#!/Dashboard/update_dashboard) + // + rpc ImportLookmlDashboard(ImportLookmlDashboardReq) returns (ImportLookmlDashboardResponse); + + // ### Update all linked dashboards to match the specified LookML dashboard. + // + // Any UDD (a dashboard which exists in the Looker database rather than as a LookML file) which has a `lookml_link_id` + // property value referring to a LookML dashboard's id (model::dashboardname) will be updated so that it matches the current state of the LookML dashboard. + // + // For this operation to succeed the user must have permission to view the LookML dashboard, and only linked dashboards + // that the user has permission to update will be synced. + // + // To **link** or **unlink** a UDD set the `lookml_link_id` property with [update_dashboard()](#!/Dashboard/update_dashboard) + // + rpc SyncLookmlDashboard(SyncLookmlDashboardReq) returns (SyncLookmlDashboardResponse); + + // ### Get information about a dashboard + // + // Returns the full details of the identified dashboard object + // + // Get a **summary list** of all active dashboards with [all_dashboards()](#!/Dashboard/all_dashboards) + // + // You can **Search** for dashboards with [search_dashboards()](#!/Dashboard/search_dashboards) + // + rpc Dashboard(DashboardReq) returns (DashboardResponse); + + // ### Update a dashboard + // + // You can use this function to change the string and integer properties of + // a dashboard. Nested objects such as filters, dashboard elements, or dashboard layout components + // cannot be modified by this function - use the update functions for the respective + // nested object types (like [update_dashboard_filter()](#!/3.1/Dashboard/update_dashboard_filter) to change a filter) + // to modify nested objects referenced by a dashboard. + // + // If you receive a 422 error response when updating a dashboard, be sure to look at the + // response body for information about exactly which fields are missing or contain invalid data. + // + rpc UpdateDashboard(UpdateDashboardReq) returns (UpdateDashboardResponse); + + // ### Delete the dashboard with the specified id + // + // Permanently **deletes** a dashboard. (The dashboard cannot be recovered after this operation.) + // + // "Soft" delete or hide a dashboard by setting its `deleted` status to `True` with [update_dashboard()](#!/Dashboard/update_dashboard). + // + // Note: When a dashboard is deleted in the UI, it is soft deleted. Use this API call to permanently remove it, if desired. + // + rpc DeleteDashboard(DeleteDashboardReq) returns (DeleteDashboardResponse); + + // ### Get Aggregate Table LookML for Each Query on a Dahboard + // + // Returns a JSON object that contains the dashboard id and Aggregate Table lookml + // + // + rpc DashboardAggregateTableLookml(DashboardAggregateTableLookmlReq) returns (DashboardAggregateTableLookmlResponse); + + // ### Get lookml of a UDD + // + // Returns a JSON object that contains the dashboard id and the full lookml + // + // + rpc DashboardLookml(DashboardLookmlReq) returns (DashboardLookmlResponse); + + // ### Move an existing dashboard + // + // Moves a dashboard to a specified folder, and returns the moved dashboard. + // + // `dashboard_id` and `folder_id` are required. + // `dashboard_id` and `folder_id` must already exist, and `folder_id` must be different from the current `folder_id` of the dashboard. + // + rpc MoveDashboard(MoveDashboardReq) returns (MoveDashboardResponse); + + // ### Creates a new dashboard object based on LookML Dashboard YAML, and returns the details of the newly created dashboard. + // + // This is equivalent to creating a LookML Dashboard and converting to a User-defined dashboard. + // + // LookML must contain valid LookML YAML code. It's recommended to use the LookML format returned + // from [dashboard_lookml()](#!/Dashboard/dashboard_lookml) as the input LookML (newlines replaced with + // ). + // + // Note that the created dashboard is not linked to any LookML Dashboard, + // i.e. [sync_lookml_dashboard()](#!/Dashboard/sync_lookml_dashboard) will not update dashboards created by this method. + // + // + rpc CreateDashboardFromLookml(CreateDashboardFromLookmlReq) returns (CreateDashboardFromLookmlResponse); + + // ### Copy an existing dashboard + // + // Creates a copy of an existing dashboard, in a specified folder, and returns the copied dashboard. + // + // `dashboard_id` is required, `dashboard_id` and `folder_id` must already exist if specified. + // `folder_id` will default to the existing folder. + // + // If a dashboard with the same title already exists in the target folder, the copy will have '(copy)' + // or '(copy <# of copies>)' appended. + // + rpc CopyDashboard(CopyDashboardReq) returns (CopyDashboardResponse); + + // ### Search Dashboard Elements + // + // Returns an **array of DashboardElement objects** that match the specified search criteria. + // + // If multiple search params are given and `filter_or` is FALSE or not specified, + // search params are combined in a logical AND operation. + // Only rows that match *all* search param criteria will be returned. + // + // If `filter_or` is TRUE, multiple search params are combined in a logical OR operation. + // Results will include rows that match **any** of the search criteria. + // + // String search params use case-insensitive matching. + // String search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions. + // example="dan%" will match "danger" and "Danzig" but not "David" + // example="D_m%" will match "Damage" and "dump" + // + // Integer search params can accept a single value or a comma separated list of values. The multiple + // values will be combined under a logical OR operation - results will match at least one of + // the given values. + // + // Most search params can accept "IS NULL" and "NOT NULL" as special expressions to match + // or exclude (respectively) rows where the column is null. + // + // Boolean search params accept only "true" and "false" as values. + // + // + rpc SearchDashboardElements(SearchDashboardElementsReq) returns (SearchDashboardElementsResponse); + + // ### Get information about the dashboard element with a specific id. + rpc DashboardElement(DashboardElementReq) returns (DashboardElementResponse); + + // ### Update the dashboard element with a specific id. + rpc UpdateDashboardElement(UpdateDashboardElementReq) returns (UpdateDashboardElementResponse); + + // ### Delete a dashboard element with a specific id. + rpc DeleteDashboardElement(DeleteDashboardElementReq) returns (DeleteDashboardElementResponse); + + // ### Get information about all the dashboard elements on a dashboard with a specific id. + rpc DashboardDashboardElements(DashboardDashboardElementsReq) returns (DashboardDashboardElementsResponse); + + // ### Create a dashboard element on the dashboard with a specific id. + rpc CreateDashboardElement(CreateDashboardElementReq) returns (CreateDashboardElementResponse); + + // ### Get information about the dashboard filters with a specific id. + rpc DashboardFilter(DashboardFilterReq) returns (DashboardFilterResponse); + + // ### Update the dashboard filter with a specific id. + rpc UpdateDashboardFilter(UpdateDashboardFilterReq) returns (UpdateDashboardFilterResponse); + + // ### Delete a dashboard filter with a specific id. + rpc DeleteDashboardFilter(DeleteDashboardFilterReq) returns (DeleteDashboardFilterResponse); + + // ### Get information about all the dashboard filters on a dashboard with a specific id. + rpc DashboardDashboardFilters(DashboardDashboardFiltersReq) returns (DashboardDashboardFiltersResponse); + + // ### Create a dashboard filter on the dashboard with a specific id. + rpc CreateDashboardFilter(CreateDashboardFilterReq) returns (CreateDashboardFilterResponse); + + // ### Get information about the dashboard elements with a specific id. + rpc DashboardLayoutComponent(DashboardLayoutComponentReq) returns (DashboardLayoutComponentResponse); + + // ### Update the dashboard element with a specific id. + rpc UpdateDashboardLayoutComponent(UpdateDashboardLayoutComponentReq) returns (UpdateDashboardLayoutComponentResponse); + + // ### Get information about all the dashboard layout components for a dashboard layout with a specific id. + rpc DashboardLayoutDashboardLayoutComponents(DashboardLayoutDashboardLayoutComponentsReq) returns (DashboardLayoutDashboardLayoutComponentsResponse); + + // ### Get information about the dashboard layouts with a specific id. + rpc DashboardLayout(DashboardLayoutReq) returns (DashboardLayoutResponse); + + // ### Update the dashboard layout with a specific id. + rpc UpdateDashboardLayout(UpdateDashboardLayoutReq) returns (UpdateDashboardLayoutResponse); + + // ### Delete a dashboard layout with a specific id. + rpc DeleteDashboardLayout(DeleteDashboardLayoutReq) returns (DeleteDashboardLayoutResponse); + + // ### Get information about all the dashboard elements on a dashboard with a specific id. + rpc DashboardDashboardLayouts(DashboardDashboardLayoutsReq) returns (DashboardDashboardLayoutsResponse); + + // ### Create a dashboard layout on the dashboard with a specific id. + rpc CreateDashboardLayout(CreateDashboardLayoutReq) returns (CreateDashboardLayoutResponse); + + + + // DataAction: Run Data Actions + + // Perform a data action. The data action object can be obtained from query results, and used to perform an arbitrary action. + rpc PerformDataAction(PerformDataActionReq) returns (PerformDataActionResponse); + + // For some data actions, the remote server may supply a form requesting further user input. This endpoint takes a data action, asks the remote server to generate a form for it, and returns that form to you for presentation to the user. + rpc FetchRemoteDataActionForm(FetchRemoteDataActionFormReq) returns (FetchRemoteDataActionFormResponse); + + + + // Datagroup: Manage Datagroups + + // ### Get information about all datagroups. + // + rpc AllDatagroups(AllDatagroupsReq) returns (AllDatagroupsResponse); + + // ### Get information about a datagroup. + // + rpc Datagroup(DatagroupReq) returns (DatagroupResponse); + + // ### Update a datagroup using the specified params. + // + rpc UpdateDatagroup(UpdateDatagroupReq) returns (UpdateDatagroupResponse); + + + + // DerivedTable: View Derived Table graphs + + // ### Discover information about derived tables + // + rpc GraphDerivedTablesForModel(GraphDerivedTablesForModelReq) returns (GraphDerivedTablesForModelResponse); + + // ### Get the subgraph representing this derived table and its dependencies. + // + rpc GraphDerivedTablesForView(GraphDerivedTablesForViewReq) returns (GraphDerivedTablesForViewResponse); + + // Enqueue materialization for a PDT with the given model name and view name + rpc StartPdtBuild(StartPdtBuildReq) returns (StartPdtBuildResponse); + + // Check status of PDT materialization + rpc CheckPdtBuild(CheckPdtBuildReq) returns (CheckPdtBuildResponse); + + // Stop a PDT materialization + rpc StopPdtBuild(StopPdtBuildReq) returns (StopPdtBuildResponse); + + + + // Folder: Manage Folders + + // Search for folders by creator id, parent id, name, etc + rpc SearchFolders(SearchFoldersReq) returns (SearchFoldersResponse); + + // ### Get information about the folder with a specific id. + rpc Folder(FolderReq) returns (FolderResponse); + + // ### Update the folder with a specific id. + rpc UpdateFolder(UpdateFolderReq) returns (UpdateFolderResponse); + + // ### Delete the folder with a specific id including any children folders. + // **DANGER** this will delete all looks and dashboards in the folder. + // + rpc DeleteFolder(DeleteFolderReq) returns (DeleteFolderResponse); + + // ### Get information about all folders. + // + // In API 3.x, this will not return empty personal folders, unless they belong to the calling user, + // or if they contain soft-deleted content. + // + // In API 4.0+, all personal folders will be returned. + // + // + rpc AllFolders(AllFoldersReq) returns (AllFoldersResponse); + + // ### Create a folder with specified information. + // + // Caller must have permission to edit the parent folder and to create folders, otherwise the request + // returns 404 Not Found. + // + rpc CreateFolder(CreateFolderReq) returns (CreateFolderResponse); + + // ### Get the children of a folder. + rpc FolderChildren(FolderChildrenReq) returns (FolderChildrenResponse); + + // ### Search the children of a folder + rpc FolderChildrenSearch(FolderChildrenSearchReq) returns (FolderChildrenSearchResponse); + + // ### Get the parent of a folder + rpc FolderParent(FolderParentReq) returns (FolderParentResponse); + + // ### Get the ancestors of a folder + rpc FolderAncestors(FolderAncestorsReq) returns (FolderAncestorsResponse); + + // ### Get all looks in a folder. + // In API 3.x, this will return all looks in a folder, including looks in the trash. + // In API 4.0+, all looks in a folder will be returned, excluding looks in the trash. + // + rpc FolderLooks(FolderLooksReq) returns (FolderLooksResponse); + + // ### Get the dashboards in a folder + rpc FolderDashboards(FolderDashboardsReq) returns (FolderDashboardsResponse); + + + + // Group: Manage Groups + + // ### Get information about all groups. + // + rpc AllGroups(AllGroupsReq) returns (AllGroupsResponse); + + // ### Creates a new group (admin only). + // + rpc CreateGroup(CreateGroupReq) returns (CreateGroupResponse); + + // ### Search groups + // + // Returns all group records that match the given search criteria. + // + // If multiple search params are given and `filter_or` is FALSE or not specified, + // search params are combined in a logical AND operation. + // Only rows that match *all* search param criteria will be returned. + // + // If `filter_or` is TRUE, multiple search params are combined in a logical OR operation. + // Results will include rows that match **any** of the search criteria. + // + // String search params use case-insensitive matching. + // String search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions. + // example="dan%" will match "danger" and "Danzig" but not "David" + // example="D_m%" will match "Damage" and "dump" + // + // Integer search params can accept a single value or a comma separated list of values. The multiple + // values will be combined under a logical OR operation - results will match at least one of + // the given values. + // + // Most search params can accept "IS NULL" and "NOT NULL" as special expressions to match + // or exclude (respectively) rows where the column is null. + // + // Boolean search params accept only "true" and "false" as values. + // + // + rpc SearchGroups(SearchGroupsReq) returns (SearchGroupsResponse); + + // ### Search groups include roles + // + // Returns all group records that match the given search criteria, and attaches any associated roles. + // + // If multiple search params are given and `filter_or` is FALSE or not specified, + // search params are combined in a logical AND operation. + // Only rows that match *all* search param criteria will be returned. + // + // If `filter_or` is TRUE, multiple search params are combined in a logical OR operation. + // Results will include rows that match **any** of the search criteria. + // + // String search params use case-insensitive matching. + // String search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions. + // example="dan%" will match "danger" and "Danzig" but not "David" + // example="D_m%" will match "Damage" and "dump" + // + // Integer search params can accept a single value or a comma separated list of values. The multiple + // values will be combined under a logical OR operation - results will match at least one of + // the given values. + // + // Most search params can accept "IS NULL" and "NOT NULL" as special expressions to match + // or exclude (respectively) rows where the column is null. + // + // Boolean search params accept only "true" and "false" as values. + // + // + rpc SearchGroupsWithRoles(SearchGroupsWithRolesReq) returns (SearchGroupsWithRolesResponse); + + // ### Search groups include hierarchy + // + // Returns all group records that match the given search criteria, and attaches + // associated role_ids and parent group_ids. + // + // If multiple search params are given and `filter_or` is FALSE or not specified, + // search params are combined in a logical AND operation. + // Only rows that match *all* search param criteria will be returned. + // + // If `filter_or` is TRUE, multiple search params are combined in a logical OR operation. + // Results will include rows that match **any** of the search criteria. + // + // String search params use case-insensitive matching. + // String search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions. + // example="dan%" will match "danger" and "Danzig" but not "David" + // example="D_m%" will match "Damage" and "dump" + // + // Integer search params can accept a single value or a comma separated list of values. The multiple + // values will be combined under a logical OR operation - results will match at least one of + // the given values. + // + // Most search params can accept "IS NULL" and "NOT NULL" as special expressions to match + // or exclude (respectively) rows where the column is null. + // + // Boolean search params accept only "true" and "false" as values. + // + // + rpc SearchGroupsWithHierarchy(SearchGroupsWithHierarchyReq) returns (SearchGroupsWithHierarchyResponse); + + // ### Get information about a group. + // + rpc Group(GroupReq) returns (GroupResponse); + + // ### Updates the a group (admin only). + rpc UpdateGroup(UpdateGroupReq) returns (UpdateGroupResponse); + + // ### Deletes a group (admin only). + // + rpc DeleteGroup(DeleteGroupReq) returns (DeleteGroupResponse); + + // ### Get information about all the groups in a group + // + rpc AllGroupGroups(AllGroupGroupsReq) returns (AllGroupGroupsResponse); + + // ### Adds a new group to a group. + // + rpc AddGroupGroup(AddGroupGroupReq) returns (AddGroupGroupResponse); + + // ### Get information about all the users directly included in a group. + // + rpc AllGroupUsers(AllGroupUsersReq) returns (AllGroupUsersResponse); + + // ### Adds a new user to a group. + // + rpc AddGroupUser(AddGroupUserReq) returns (AddGroupUserResponse); + + // ### Removes a user from a group. + // + rpc DeleteGroupUser(DeleteGroupUserReq) returns (DeleteGroupUserResponse); + + // ### Removes a group from a group. + // + rpc DeleteGroupFromGroup(DeleteGroupFromGroupReq) returns (DeleteGroupFromGroupResponse); + + // ### Set the value of a user attribute for a group. + // + // For information about how user attribute values are calculated, see [Set User Attribute Group Values](#!/UserAttribute/set_user_attribute_group_values). + // + rpc UpdateUserAttributeGroupValue(UpdateUserAttributeGroupValueReq) returns (UpdateUserAttributeGroupValueResponse); + + // ### Remove a user attribute value from a group. + // + rpc DeleteUserAttributeGroupValue(DeleteUserAttributeGroupValueReq) returns (DeleteUserAttributeGroupValueResponse); + + + + // Homepage: Manage Homepage + + // ### Get information about the primary homepage's sections. + // + rpc AllPrimaryHomepageSections(AllPrimaryHomepageSectionsReq) returns (AllPrimaryHomepageSectionsResponse); + + + + // Integration: Manage Integrations + + // ### Get information about all Integration Hubs. + // + rpc AllIntegrationHubs(AllIntegrationHubsReq) returns (AllIntegrationHubsResponse); + + // ### Create a new Integration Hub. + // + // This API is rate limited to prevent it from being used for SSRF attacks + // + rpc CreateIntegrationHub(CreateIntegrationHubReq) returns (CreateIntegrationHubResponse); + + // ### Get information about a Integration Hub. + // + rpc IntegrationHub(IntegrationHubReq) returns (IntegrationHubResponse); + + // ### Update a Integration Hub definition. + // + // This API is rate limited to prevent it from being used for SSRF attacks + // + rpc UpdateIntegrationHub(UpdateIntegrationHubReq) returns (UpdateIntegrationHubResponse); + + // ### Delete a Integration Hub. + // + rpc DeleteIntegrationHub(DeleteIntegrationHubReq) returns (DeleteIntegrationHubResponse); + + // Accepts the legal agreement for a given integration hub. This only works for integration hubs that have legal_agreement_required set to true and legal_agreement_signed set to false. + rpc AcceptIntegrationHubLegalAgreement(AcceptIntegrationHubLegalAgreementReq) returns (AcceptIntegrationHubLegalAgreementResponse); + + // ### Get information about all Integrations. + // + rpc AllIntegrations(AllIntegrationsReq) returns (AllIntegrationsResponse); + + // ### Get information about a Integration. + // + rpc Integration(IntegrationReq) returns (IntegrationResponse); + + // ### Update parameters on a Integration. + // + rpc UpdateIntegration(UpdateIntegrationReq) returns (UpdateIntegrationResponse); + + // Returns the Integration form for presentation to the user. + rpc FetchIntegrationForm(FetchIntegrationFormReq) returns (FetchIntegrationFormResponse); + + // Tests the integration to make sure all the settings are working. + rpc TestIntegration(TestIntegrationReq) returns (TestIntegrationResponse); + + + + // Look: Run and Manage Looks + + // ### Get information about all active Looks + // + // Returns an array of **abbreviated Look objects** describing all the looks that the caller has access to. Soft-deleted Looks are **not** included. + // + // Get the **full details** of a specific look by id with [look(id)](#!/Look/look) + // + // Find **soft-deleted looks** with [search_looks()](#!/Look/search_looks) + // + rpc AllLooks(AllLooksReq) returns (AllLooksResponse); + + // ### Create a Look + // + // To create a look to display query data, first create the query with [create_query()](#!/Query/create_query) + // then assign the query's id to the `query_id` property in the call to `create_look()`. + // + // To place the look into a particular space, assign the space's id to the `space_id` property + // in the call to `create_look()`. + // + rpc CreateLook(CreateLookReq) returns (CreateLookResponse); + + // ### Search Looks + // + // Returns an **array of Look objects** that match the specified search criteria. + // + // If multiple search params are given and `filter_or` is FALSE or not specified, + // search params are combined in a logical AND operation. + // Only rows that match *all* search param criteria will be returned. + // + // If `filter_or` is TRUE, multiple search params are combined in a logical OR operation. + // Results will include rows that match **any** of the search criteria. + // + // String search params use case-insensitive matching. + // String search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions. + // example="dan%" will match "danger" and "Danzig" but not "David" + // example="D_m%" will match "Damage" and "dump" + // + // Integer search params can accept a single value or a comma separated list of values. The multiple + // values will be combined under a logical OR operation - results will match at least one of + // the given values. + // + // Most search params can accept "IS NULL" and "NOT NULL" as special expressions to match + // or exclude (respectively) rows where the column is null. + // + // Boolean search params accept only "true" and "false" as values. + // + // + // Get a **single look** by id with [look(id)](#!/Look/look) + // + rpc SearchLooks(SearchLooksReq) returns (SearchLooksResponse); + + // ### Get a Look. + // + // Returns detailed information about a Look and its associated Query. + // + // + rpc Look(LookReq) returns (LookResponse); + + // ### Modify a Look + // + // Use this function to modify parts of a look. Property values given in a call to `update_look` are + // applied to the existing look, so there's no need to include properties whose values are not changing. + // It's best to specify only the properties you want to change and leave everything else out + // of your `update_look` call. **Look properties marked 'read-only' will be ignored.** + // + // When a user deletes a look in the Looker UI, the look data remains in the database but is + // marked with a deleted flag ("soft-deleted"). Soft-deleted looks can be undeleted (by an admin) + // if the delete was in error. + // + // To soft-delete a look via the API, use [update_look()](#!/Look/update_look) to change the look's `deleted` property to `true`. + // You can undelete a look by calling `update_look` to change the look's `deleted` property to `false`. + // + // Soft-deleted looks are excluded from the results of [all_looks()](#!/Look/all_looks) and [search_looks()](#!/Look/search_looks), so they + // essentially disappear from view even though they still reside in the db. + // In API 3.1 and later, you can pass `deleted: true` as a parameter to [search_looks()](#!/3.1/Look/search_looks) to list soft-deleted looks. + // + // NOTE: [delete_look()](#!/Look/delete_look) performs a "hard delete" - the look data is removed from the Looker + // database and destroyed. There is no "undo" for `delete_look()`. + // + rpc UpdateLook(UpdateLookReq) returns (UpdateLookResponse); + + // ### Permanently Delete a Look + // + // This operation **permanently** removes a look from the Looker database. + // + // NOTE: There is no "undo" for this kind of delete. + // + // For information about soft-delete (which can be undone) see [update_look()](#!/Look/update_look). + // + rpc DeleteLook(DeleteLookReq) returns (DeleteLookResponse); + + // ### Run a Look + // + // Runs a given look's query and returns the results in the requested format. + // + // Supported formats: + // + // | result_format | Description + // | :-----------: | :--- | + // | json | Plain json + // | json_detail | Row data plus metadata describing the fields, pivots, table calcs, and other aspects of the query + // | csv | Comma separated values with a header + // | txt | Tab separated values with a header + // | html | Simple html + // | md | Simple markdown + // | xlsx | MS Excel spreadsheet + // | sql | Returns the generated SQL rather than running the query + // | png | A PNG image of the visualization of the query + // | jpg | A JPG image of the visualization of the query + // + // + // + rpc RunLook(RunLookReq) returns (RunLookResponse); + + // ### Copy an existing look + // + // Creates a copy of an existing look, in a specified folder, and returns the copied look. + // + // `look_id` and `folder_id` are required. + // + // `look_id` and `folder_id` must already exist, and `folder_id` must be different from the current `folder_id` of the dashboard. + // + rpc CopyLook(CopyLookReq) returns (CopyLookResponse); + + // ### Move an existing look + // + // Moves a look to a specified folder, and returns the moved look. + // + // `look_id` and `folder_id` are required. + // `look_id` and `folder_id` must already exist, and `folder_id` must be different from the current `folder_id` of the dashboard. + // + rpc MoveLook(MoveLookReq) returns (MoveLookResponse); + + + + // LookmlModel: Manage LookML Models + + // ### Get information about all lookml models. + // + rpc AllLookmlModels(AllLookmlModelsReq) returns (AllLookmlModelsResponse); + + // ### Create a lookml model using the specified configuration. + // + rpc CreateLookmlModel(CreateLookmlModelReq) returns (CreateLookmlModelResponse); + + // ### Get information about a lookml model. + // + rpc LookmlModel(LookmlModelReq) returns (LookmlModelResponse); + + // ### Update a lookml model using the specified configuration. + // + rpc UpdateLookmlModel(UpdateLookmlModelReq) returns (UpdateLookmlModelResponse); + + // ### Delete a lookml model. + // + rpc DeleteLookmlModel(DeleteLookmlModelReq) returns (DeleteLookmlModelResponse); + + // ### Get information about a lookml model explore. + // + rpc LookmlModelExplore(LookmlModelExploreReq) returns (LookmlModelExploreResponse); + + + + // Metadata: Connection Metadata Features + + // ### Field name suggestions for a model and view + // + // `filters` is a string hash of values, with the key as the field name and the string value as the filter expression: + // + // ```ruby + // {'users.age': '>=60'} + // ``` + // + // or + // + // ```ruby + // {'users.age': '<30'} + // ``` + // + // or + // + // ```ruby + // {'users.age': '=50'} + // ``` + // + rpc ModelFieldnameSuggestions(ModelFieldnameSuggestionsReq) returns (ModelFieldnameSuggestionsResponse); + + // ### Get a single model + // + // + rpc GetModel(GetModelReq) returns (GetModelResponse); + + // ### List databases available to this connection + // + // Certain dialects can support multiple databases per single connection. + // If this connection supports multiple databases, the database names will be returned in an array. + // + // Connections using dialects that do not support multiple databases will return an empty array. + // + // **Note**: [Connection Features](#!/Metadata/connection_features) can be used to determine if a connection supports + // multiple databases. + // + rpc ConnectionDatabases(ConnectionDatabasesReq) returns (ConnectionDatabasesResponse); + + // ### Retrieve metadata features for this connection + // + // Returns a list of feature names with `true` (available) or `false` (not available) + // + // + rpc ConnectionFeatures(ConnectionFeaturesReq) returns (ConnectionFeaturesResponse); + + // ### Get the list of schemas and tables for a connection + // + // + rpc ConnectionSchemas(ConnectionSchemasReq) returns (ConnectionSchemasResponse); + + // ### Get the list of tables for a schema + // + // For dialects that support multiple databases, optionally identify which to use. If not provided, the default + // database for the connection will be used. + // + // For dialects that do **not** support multiple databases, **do not use** the database parameter + // + rpc ConnectionTables(ConnectionTablesReq) returns (ConnectionTablesResponse); + + // ### Get the columns (and therefore also the tables) in a specific schema + // + // + rpc ConnectionColumns(ConnectionColumnsReq) returns (ConnectionColumnsResponse); + + // ### Search a connection for columns matching the specified name + // + // **Note**: `column_name` must be a valid column name. It is not a search pattern. + // + rpc ConnectionSearchColumns(ConnectionSearchColumnsReq) returns (ConnectionSearchColumnsResponse); + + // ### Connection cost estimating + // + // Assign a `sql` statement to the body of the request. e.g., for Ruby, `{sql: 'select * from users'}` + // + // **Note**: If the connection's dialect has no support for cost estimates, an error will be returned + // + rpc ConnectionCostEstimate(ConnectionCostEstimateReq) returns (ConnectionCostEstimateResponse); + + + + // Project: Manage Projects + + // ### Generate Lockfile for All LookML Dependencies + // + // Git must have been configured, must be in dev mode and deploy permission required + // + // Install_all is a two step process + // 1. For each remote_dependency in a project the dependency manager will resolve any ambiguous ref. + // 2. The project will then write out a lockfile including each remote_dependency with its resolved ref. + // + // + rpc LockAll(LockAllReq) returns (LockAllResponse); + + // ### Get All Git Branches + // + // Returns a list of git branches in the project repository + // + rpc AllGitBranches(AllGitBranchesReq) returns (AllGitBranchesResponse); + + // ### Get the Current Git Branch + // + // Returns the git branch currently checked out in the given project repository + // + rpc GitBranch(GitBranchReq) returns (GitBranchResponse); + + // ### Checkout and/or reset --hard an existing Git Branch + // + // Only allowed in development mode + // - Call `update_session` to select the 'dev' workspace. + // + // Checkout an existing branch if name field is different from the name of the currently checked out branch. + // + // Optionally specify a branch name, tag name or commit SHA to which the branch should be reset. + // **DANGER** hard reset will be force pushed to the remote. Unsaved changes and commits may be permanently lost. + // + // + rpc UpdateGitBranch(UpdateGitBranchReq) returns (UpdateGitBranchResponse); + + // ### Create and Checkout a Git Branch + // + // Creates and checks out a new branch in the given project repository + // Only allowed in development mode + // - Call `update_session` to select the 'dev' workspace. + // + // Optionally specify a branch name, tag name or commit SHA as the start point in the ref field. + // If no ref is specified, HEAD of the current branch will be used as the start point for the new branch. + // + // + rpc CreateGitBranch(CreateGitBranchReq) returns (CreateGitBranchResponse); + + // ### Get the specified Git Branch + // + // Returns the git branch specified in branch_name path param if it exists in the given project repository + // + rpc FindGitBranch(FindGitBranchReq) returns (FindGitBranchResponse); + + // ### Delete the specified Git Branch + // + // Delete git branch specified in branch_name path param from local and remote of specified project repository + // + rpc DeleteGitBranch(DeleteGitBranchReq) returns (DeleteGitBranchResponse); + + // ### Deploy a Remote Branch or Ref to Production + // + // Git must have been configured and deploy permission required. + // + // Deploy is a one/two step process + // 1. If this is the first deploy of this project, create the production project with git repository. + // 2. Pull the branch or ref into the production project. + // + // Can only specify either a branch or a ref. + // + // + rpc DeployRefToProduction(DeployRefToProductionReq) returns (DeployRefToProductionResponse); + + // ### Deploy LookML from this Development Mode Project to Production + // + // Git must have been configured, must be in dev mode and deploy permission required + // + // Deploy is a two / three step process: + // + // 1. Push commits in current branch of dev mode project to the production branch (origin/master). + // Note a. This step is skipped in read-only projects. + // Note b. If this step is unsuccessful for any reason (e.g. rejected non-fastforward because production branch has + // commits not in current branch), subsequent steps will be skipped. + // 2. If this is the first deploy of this project, create the production project with git repository. + // 3. Pull the production branch into the production project. + // + // + rpc DeployToProduction(DeployToProductionReq) returns (DeployToProductionResponse); + + // ### Reset a project to the revision of the project that is in production. + // + // **DANGER** this will delete any changes that have not been pushed to a remote repository. + // + rpc ResetProjectToProduction(ResetProjectToProductionReq) returns (ResetProjectToProductionResponse); + + // ### Reset a project development branch to the revision of the project that is on the remote. + // + // **DANGER** this will delete any changes that have not been pushed to a remote repository. + // + rpc ResetProjectToRemote(ResetProjectToRemoteReq) returns (ResetProjectToRemoteResponse); + + // ### Get All Projects + // + // Returns all projects visible to the current user + // + rpc AllProjects(AllProjectsReq) returns (AllProjectsResponse); + + // ### Create A Project + // + // dev mode required. + // - Call `update_session` to select the 'dev' workspace. + // + // `name` is required. + // `git_remote_url` is not allowed. To configure Git for the newly created project, follow the instructions in `update_project`. + // + // + rpc CreateProject(CreateProjectReq) returns (CreateProjectResponse); + + // ### Get A Project + // + // Returns the project with the given project id + // + rpc Project(ProjectReq) returns (ProjectResponse); + + // ### Update Project Configuration + // + // Apply changes to a project's configuration. + // + // + // #### Configuring Git for a Project + // + // To set up a Looker project with a remote git repository, follow these steps: + // + // 1. Call `update_session` to select the 'dev' workspace. + // 1. Call `create_git_deploy_key` to create a new deploy key for the project + // 1. Copy the deploy key text into the remote git repository's ssh key configuration + // 1. Call `update_project` to set project's `git_remote_url` ()and `git_service_name`, if necessary). + // + // When you modify a project's `git_remote_url`, Looker connects to the remote repository to fetch + // metadata. The remote git repository MUST be configured with the Looker-generated deploy + // key for this project prior to setting the project's `git_remote_url`. + // + // To set up a Looker project with a git repository residing on the Looker server (a 'bare' git repo): + // + // 1. Call `update_session` to select the 'dev' workspace. + // 1. Call `update_project` setting `git_remote_url` to null and `git_service_name` to "bare". + // + // + rpc UpdateProject(UpdateProjectReq) returns (UpdateProjectResponse); + + // ### Get A Projects Manifest object + // + // Returns the project with the given project id + // + rpc Manifest(ManifestReq) returns (ManifestResponse); + + // ### Git Deploy Key + // + // Returns the ssh public key previously created for a project's git repository. + // + rpc GitDeployKey(GitDeployKeyReq) returns (GitDeployKeyResponse); + + // ### Create Git Deploy Key + // + // Create a public/private key pair for authenticating ssh git requests from Looker to a remote git repository + // for a particular Looker project. + // + // Returns the public key of the generated ssh key pair. + // + // Copy this public key to your remote git repository's ssh keys configuration so that the remote git service can + // validate and accept git requests from the Looker server. + // + rpc CreateGitDeployKey(CreateGitDeployKeyReq) returns (CreateGitDeployKeyResponse); + + // ### Get Cached Project Validation Results + // + // Returns the cached results of a previous project validation calculation, if any. + // Returns http status 204 No Content if no validation results exist. + // + // Validating the content of all the files in a project can be computationally intensive + // for large projects. Use this API to simply fetch the results of the most recent + // project validation rather than revalidating the entire project from scratch. + // + // A value of `"stale": true` in the response indicates that the project has changed since + // the cached validation results were computed. The cached validation results may no longer + // reflect the current state of the project. + // + rpc ProjectValidationResults(ProjectValidationResultsReq) returns (ProjectValidationResultsResponse); + + // ### Validate Project + // + // Performs lint validation of all lookml files in the project. + // Returns a list of errors found, if any. + // + // Validating the content of all the files in a project can be computationally intensive + // for large projects. For best performance, call `validate_project(project_id)` only + // when you really want to recompute project validation. To quickly display the results of + // the most recent project validation (without recomputing), use `project_validation_results(project_id)` + // + rpc ValidateProject(ValidateProjectReq) returns (ValidateProjectResponse); + + // ### Get Project Workspace + // + // Returns information about the state of the project files in the currently selected workspace + // + rpc ProjectWorkspace(ProjectWorkspaceReq) returns (ProjectWorkspaceResponse); + + // ### Get All Project Files + // + // Returns a list of the files in the project + // + rpc AllProjectFiles(AllProjectFilesReq) returns (AllProjectFilesResponse); + + // ### Get Project File Info + // + // Returns information about a file in the project + // + rpc ProjectFile(ProjectFileReq) returns (ProjectFileResponse); + + // ### Get All Git Connection Tests + // + // dev mode required. + // - Call `update_session` to select the 'dev' workspace. + // + // Returns a list of tests which can be run against a project's (or the dependency project for the provided remote_url) git connection. Call [Run Git Connection Test](#!/Project/run_git_connection_test) to execute each test in sequence. + // + // Tests are ordered by increasing specificity. Tests should be run in the order returned because later tests require functionality tested by tests earlier in the test list. + // + // For example, a late-stage test for write access is meaningless if connecting to the git server (an early test) is failing. + // + rpc AllGitConnectionTests(AllGitConnectionTestsReq) returns (AllGitConnectionTestsResponse); + + // ### Run a git connection test + // + // Run the named test on the git service used by this project (or the dependency project for the provided remote_url) and return the result. This + // is intended to help debug git connections when things do not work properly, to give + // more helpful information about why a git url is not working with Looker. + // + // Tests should be run in the order they are returned by [Get All Git Connection Tests](#!/Project/all_git_connection_tests). + // + rpc RunGitConnectionTest(RunGitConnectionTestReq) returns (RunGitConnectionTestResponse); + + // ### Get All LookML Tests + // + // Returns a list of tests which can be run to validate a project's LookML code and/or the underlying data, + // optionally filtered by the file id. + // Call [Run LookML Test](#!/Project/run_lookml_test) to execute tests. + // + rpc AllLookmlTests(AllLookmlTestsReq) returns (AllLookmlTestsResponse); + + // ### Run LookML Tests + // + // Runs all tests in the project, optionally filtered by file, test, and/or model. + // + rpc RunLookmlTest(RunLookmlTestReq) returns (RunLookmlTestResponse); + + // ### Creates a tag for the most recent commit, or a specific ref is a SHA is provided + // + // This is an internal-only, undocumented route. + // + rpc TagRef(TagRefReq) returns (TagRefResponse); + + // ### Configure Repository Credential for a remote dependency + // + // Admin required. + // + // `root_project_id` is required. + // `credential_id` is required. + // + // + rpc UpdateRepositoryCredential(UpdateRepositoryCredentialReq) returns (UpdateRepositoryCredentialResponse); + + // ### Repository Credential for a remote dependency + // + // Admin required. + // + // `root_project_id` is required. + // `credential_id` is required. + // + rpc DeleteRepositoryCredential(DeleteRepositoryCredentialReq) returns (DeleteRepositoryCredentialResponse); + + // ### Get all Repository Credentials for a project + // + // `root_project_id` is required. + // + rpc GetAllRepositoryCredentials(GetAllRepositoryCredentialsReq) returns (GetAllRepositoryCredentialsResponse); + + + + // Query: Run and Manage Queries + + // ### Create an async query task + // + // Creates a query task (job) to run a previously created query asynchronously. Returns a Query Task ID. + // + // Use [query_task(query_task_id)](#!/Query/query_task) to check the execution status of the query task. + // After the query task status reaches "Complete", use [query_task_results(query_task_id)](#!/Query/query_task_results) to fetch the results of the query. + // + rpc CreateQueryTask(CreateQueryTaskReq) returns (CreateQueryTaskResponse); + + // ### Fetch results of multiple async queries + // + // Returns the results of multiple async queries in one request. + // + // For Query Tasks that are not completed, the response will include the execution status of the Query Task but will not include query results. + // Query Tasks whose results have expired will have a status of 'expired'. + // If the user making the API request does not have sufficient privileges to view a Query Task result, the result will have a status of 'missing' + // + rpc QueryTaskMultiResults(QueryTaskMultiResultsReq) returns (QueryTaskMultiResultsResponse); + + // ### Get Query Task details + // + // Use this function to check the status of an async query task. After the status + // reaches "Complete", you can call [query_task_results(query_task_id)](#!/Query/query_task_results) to + // retrieve the results of the query. + // + // Use [create_query_task()](#!/Query/create_query_task) to create an async query task. + // + rpc QueryTask(QueryTaskReq) returns (QueryTaskResponse); + + // ### Get Async Query Results + // + // Returns the results of an async query task if the query has completed. + // + // If the query task is still running or waiting to run, this function returns 204 No Content. + // + // If the query task ID is invalid or the cached results of the query task have expired, this function returns 404 Not Found. + // + // Use [query_task(query_task_id)](#!/Query/query_task) to check the execution status of the query task + // Call query_task_results only after the query task status reaches "Complete". + // + // You can also use [query_task_multi_results()](#!/Query/query_task_multi_results) retrieve the + // results of multiple async query tasks at the same time. + // + // #### SQL Error Handling: + // If the query fails due to a SQL db error, how this is communicated depends on the result_format you requested in `create_query_task()`. + // + // For `json_detail` result_format: `query_task_results()` will respond with HTTP status '200 OK' and db SQL error info + // will be in the `errors` property of the response object. The 'data' property will be empty. + // + // For all other result formats: `query_task_results()` will respond with HTTP status `400 Bad Request` and some db SQL error info + // will be in the message of the 400 error response, but not as detailed as expressed in `json_detail.errors`. + // These data formats can only carry row data, and error info is not row data. + // + rpc QueryTaskResults(QueryTaskResultsReq) returns (QueryTaskResultsResponse); + + // ### Get a previously created query by id. + // + // A Looker query object includes the various parameters that define a database query that has been run or + // could be run in the future. These parameters include: model, view, fields, filters, pivots, etc. + // Query *results* are not part of the query object. + // + // Query objects are unique and immutable. Query objects are created automatically in Looker as users explore data. + // Looker does not delete them; they become part of the query history. When asked to create a query for + // any given set of parameters, Looker will first try to find an existing query object with matching + // parameters and will only create a new object when an appropriate object can not be found. + // + // This 'get' method is used to get the details about a query for a given id. See the other methods here + // to 'create' and 'run' queries. + // + // Note that some fields like 'filter_config' and 'vis_config' etc are specific to how the Looker UI + // builds queries and visualizations and are not generally useful for API use. They are not required when + // creating new queries and can usually just be ignored. + // + // + rpc Query(QueryReq) returns (QueryResponse); + + // ### Get the query for a given query slug. + // + // This returns the query for the 'slug' in a query share URL. + // + // The 'slug' is a randomly chosen short string that is used as an alternative to the query's id value + // for use in URLs etc. This method exists as a convenience to help you use the API to 'find' queries that + // have been created using the Looker UI. + // + // You can use the Looker explore page to build a query and then choose the 'Share' option to + // show the share url for the query. Share urls generally look something like 'https://looker.yourcompany/x/vwGSbfc'. + // The trailing 'vwGSbfc' is the share slug. You can pass that string to this api method to get details about the query. + // Those details include the 'id' that you can use to run the query. Or, you can copy the query body + // (perhaps with your own modification) and use that as the basis to make/run new queries. + // + // This will also work with slugs from Looker explore urls like + // 'https://looker.yourcompany/explore/ecommerce/orders?qid=aogBgL6o3cKK1jN3RoZl5s'. In this case + // 'aogBgL6o3cKK1jN3RoZl5s' is the slug. + // + rpc QueryForSlug(QueryForSlugReq) returns (QueryForSlugResponse); + + // ### Create a query. + // + // This allows you to create a new query that you can later run. Looker queries are immutable once created + // and are not deleted. If you create a query that is exactly like an existing query then the existing query + // will be returned and no new query will be created. Whether a new query is created or not, you can use + // the 'id' in the returned query with the 'run' method. + // + // The query parameters are passed as json in the body of the request. + // + // + rpc CreateQuery(CreateQueryReq) returns (CreateQueryResponse); + + // ### Run a saved query. + // + // This runs a previously saved query. You can use this on a query that was generated in the Looker UI + // or one that you have explicitly created using the API. You can also use a query 'id' from a saved 'Look'. + // + // The 'result_format' parameter specifies the desired structure and format of the response. + // + // Supported formats: + // + // | result_format | Description + // | :-----------: | :--- | + // | json | Plain json + // | json_detail | Row data plus metadata describing the fields, pivots, table calcs, and other aspects of the query + // | csv | Comma separated values with a header + // | txt | Tab separated values with a header + // | html | Simple html + // | md | Simple markdown + // | xlsx | MS Excel spreadsheet + // | sql | Returns the generated SQL rather than running the query + // | png | A PNG image of the visualization of the query + // | jpg | A JPG image of the visualization of the query + // + // + // + rpc RunQuery(RunQueryReq) returns (RunQueryResponse); + + // ### Run the query that is specified inline in the posted body. + // + // This allows running a query as defined in json in the posted body. This combines + // the two actions of posting & running a query into one step. + // + // Here is an example body in json: + // ``` + // { + // "model":"thelook", + // "view":"inventory_items", + // "fields":["category.name","inventory_items.days_in_inventory_tier","products.count"], + // "filters":{"category.name":"socks"}, + // "sorts":["products.count desc 0"], + // "limit":"500", + // "query_timezone":"America/Los_Angeles" + // } + // ``` + // + // When using the Ruby SDK this would be passed as a Ruby hash like: + // ``` + // { + // :model=>"thelook", + // :view=>"inventory_items", + // :fields=> + // ["category.name", + // "inventory_items.days_in_inventory_tier", + // "products.count"], + // :filters=>{:"category.name"=>"socks"}, + // :sorts=>["products.count desc 0"], + // :limit=>"500", + // :query_timezone=>"America/Los_Angeles", + // } + // ``` + // + // This will return the result of running the query in the format specified by the 'result_format' parameter. + // + // Supported formats: + // + // | result_format | Description + // | :-----------: | :--- | + // | json | Plain json + // | json_detail | Row data plus metadata describing the fields, pivots, table calcs, and other aspects of the query + // | csv | Comma separated values with a header + // | txt | Tab separated values with a header + // | html | Simple html + // | md | Simple markdown + // | xlsx | MS Excel spreadsheet + // | sql | Returns the generated SQL rather than running the query + // | png | A PNG image of the visualization of the query + // | jpg | A JPG image of the visualization of the query + // + // + // + rpc RunInlineQuery(RunInlineQueryReq) returns (RunInlineQueryResponse); + + // ### Run an URL encoded query. + // + // This requires the caller to encode the specifiers for the query into the URL query part using + // Looker-specific syntax as explained below. + // + // Generally, you would want to use one of the methods that takes the parameters as json in the POST body + // for creating and/or running queries. This method exists for cases where one really needs to encode the + // parameters into the URL of a single 'GET' request. This matches the way that the Looker UI formats + // 'explore' URLs etc. + // + // The parameters here are very similar to the json body formatting except that the filter syntax is + // tricky. Unfortunately, this format makes this method not currently callable via the 'Try it out!' button + // in this documentation page. But, this is callable when creating URLs manually or when using the Looker SDK. + // + // Here is an example inline query URL: + // + // ``` + // https://looker.mycompany.com:19999/api/3.0/queries/models/thelook/views/inventory_items/run/json?fields=category.name,inventory_items.days_in_inventory_tier,products.count&f[category.name]=socks&sorts=products.count+desc+0&limit=500&query_timezone=America/Los_Angeles + // ``` + // + // When invoking this endpoint with the Ruby SDK, pass the query parameter parts as a hash. The hash to match the above would look like: + // + // ```ruby + // query_params = + // { + // fields: "category.name,inventory_items.days_in_inventory_tier,products.count", + // :"f[category.name]" => "socks", + // sorts: "products.count desc 0", + // limit: "500", + // query_timezone: "America/Los_Angeles" + // } + // response = ruby_sdk.run_url_encoded_query('thelook','inventory_items','json', query_params) + // + // ``` + // + // Again, it is generally easier to use the variant of this method that passes the full query in the POST body. + // This method is available for cases where other alternatives won't fit the need. + // + // Supported formats: + // + // | result_format | Description + // | :-----------: | :--- | + // | json | Plain json + // | json_detail | Row data plus metadata describing the fields, pivots, table calcs, and other aspects of the query + // | csv | Comma separated values with a header + // | txt | Tab separated values with a header + // | html | Simple html + // | md | Simple markdown + // | xlsx | MS Excel spreadsheet + // | sql | Returns the generated SQL rather than running the query + // | png | A PNG image of the visualization of the query + // | jpg | A JPG image of the visualization of the query + // + // + // + rpc RunUrlEncodedQuery(RunUrlEncodedQueryReq) returns (RunUrlEncodedQueryResponse); + + // ### Get Merge Query + // + // Returns a merge query object given its id. + // + rpc MergeQuery(MergeQueryReq) returns (MergeQueryResponse); + + // ### Create Merge Query + // + // Creates a new merge query object. + // + // A merge query takes the results of one or more queries and combines (merges) the results + // according to field mapping definitions. The result is similar to a SQL left outer join. + // + // A merge query can merge results of queries from different SQL databases. + // + // The order that queries are defined in the source_queries array property is significant. The + // first query in the array defines the primary key into which the results of subsequent + // queries will be merged. + // + // Like model/view query objects, merge queries are immutable and have structural identity - if + // you make a request to create a new merge query that is identical to an existing merge query, + // the existing merge query will be returned instead of creating a duplicate. Conversely, any + // change to the contents of a merge query will produce a new object with a new id. + // + rpc CreateMergeQuery(CreateMergeQueryReq) returns (CreateMergeQueryResponse); + + // Get information about all running queries. + // + rpc AllRunningQueries(AllRunningQueriesReq) returns (AllRunningQueriesResponse); + + // Kill a query with a specific query_task_id. + // + rpc KillQuery(KillQueryReq) returns (KillQueryResponse); + + // Get a SQL Runner query. + rpc SqlQuery(SqlQueryReq) returns (SqlQueryResponse); + + // ### Create a SQL Runner Query + // + // Either the `connection_name` or `model_name` parameter MUST be provided. + // + rpc CreateSqlQuery(CreateSqlQueryReq) returns (CreateSqlQueryResponse); + + // Execute a SQL Runner query in a given result_format. + rpc RunSqlQuery(RunSqlQueryReq) returns (RunSqlQueryResponse); + + + + // RenderTask: Manage Render Tasks + + // ### Create a new task to render a look to an image. + // + // Returns a render task object. + // To check the status of a render task, pass the render_task.id to [Get Render Task](#!/RenderTask/get_render_task). + // Once the render task is complete, you can download the resulting document or image using [Get Render Task Results](#!/RenderTask/get_render_task_results). + // + // + rpc CreateLookRenderTask(CreateLookRenderTaskReq) returns (CreateLookRenderTaskResponse); + + // ### Create a new task to render an existing query to an image. + // + // Returns a render task object. + // To check the status of a render task, pass the render_task.id to [Get Render Task](#!/RenderTask/get_render_task). + // Once the render task is complete, you can download the resulting document or image using [Get Render Task Results](#!/RenderTask/get_render_task_results). + // + // + rpc CreateQueryRenderTask(CreateQueryRenderTaskReq) returns (CreateQueryRenderTaskResponse); + + // ### Create a new task to render a dashboard to a document or image. + // + // Returns a render task object. + // To check the status of a render task, pass the render_task.id to [Get Render Task](#!/RenderTask/get_render_task). + // Once the render task is complete, you can download the resulting document or image using [Get Render Task Results](#!/RenderTask/get_render_task_results). + // + // + rpc CreateDashboardRenderTask(CreateDashboardRenderTaskReq) returns (CreateDashboardRenderTaskResponse); + + // ### Get information about a render task. + // + // Returns a render task object. + // To check the status of a render task, pass the render_task.id to [Get Render Task](#!/RenderTask/get_render_task). + // Once the render task is complete, you can download the resulting document or image using [Get Render Task Results](#!/RenderTask/get_render_task_results). + // + // + rpc RenderTask(RenderTaskReq) returns (RenderTaskResponse); + + // ### Get the document or image produced by a completed render task. + // + // Note that the PDF or image result will be a binary blob in the HTTP response, as indicated by the + // Content-Type in the response headers. This may require specialized (or at least different) handling than text + // responses such as JSON. You may need to tell your HTTP client that the response is binary so that it does not + // attempt to parse the binary data as text. + // + // If the render task exists but has not finished rendering the results, the response HTTP status will be + // **202 Accepted**, the response body will be empty, and the response will have a Retry-After header indicating + // that the caller should repeat the request at a later time. + // + // Returns 404 if the render task cannot be found, if the cached result has expired, or if the caller + // does not have permission to view the results. + // + // For detailed information about the status of the render task, use [Render Task](#!/RenderTask/render_task). + // Polling loops waiting for completion of a render task would be better served by polling **render_task(id)** until + // the task status reaches completion (or error) instead of polling **render_task_results(id)** alone. + // + rpc RenderTaskResults(RenderTaskResultsReq) returns (RenderTaskResultsResponse); + + // ### Create a new task to render a dashboard element to an image. + // + // Returns a render task object. + // To check the status of a render task, pass the render_task.id to [Get Render Task](#!/RenderTask/get_render_task). + // Once the render task is complete, you can download the resulting document or image using [Get Render Task Results](#!/RenderTask/get_render_task_results). + // + // + rpc CreateDashboardElementRenderTask(CreateDashboardElementRenderTaskReq) returns (CreateDashboardElementRenderTaskResponse); + + + + // Role: Manage Roles + + // ### Search model sets + // Returns all model set records that match the given search criteria. + // If multiple search params are given and `filter_or` is FALSE or not specified, + // search params are combined in a logical AND operation. + // Only rows that match *all* search param criteria will be returned. + // + // If `filter_or` is TRUE, multiple search params are combined in a logical OR operation. + // Results will include rows that match **any** of the search criteria. + // + // String search params use case-insensitive matching. + // String search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions. + // example="dan%" will match "danger" and "Danzig" but not "David" + // example="D_m%" will match "Damage" and "dump" + // + // Integer search params can accept a single value or a comma separated list of values. The multiple + // values will be combined under a logical OR operation - results will match at least one of + // the given values. + // + // Most search params can accept "IS NULL" and "NOT NULL" as special expressions to match + // or exclude (respectively) rows where the column is null. + // + // Boolean search params accept only "true" and "false" as values. + // + // + rpc SearchModelSets(SearchModelSetsReq) returns (SearchModelSetsResponse); + + // ### Get information about the model set with a specific id. + // + rpc ModelSet(ModelSetReq) returns (ModelSetResponse); + + // ### Update information about the model set with a specific id. + // + rpc UpdateModelSet(UpdateModelSetReq) returns (UpdateModelSetResponse); + + // ### Delete the model set with a specific id. + // + rpc DeleteModelSet(DeleteModelSetReq) returns (DeleteModelSetResponse); + + // ### Get information about all model sets. + // + rpc AllModelSets(AllModelSetsReq) returns (AllModelSetsResponse); + + // ### Create a model set with the specified information. Model sets are used by Roles. + // + rpc CreateModelSet(CreateModelSetReq) returns (CreateModelSetResponse); + + // ### Get all supported permissions. + // + rpc AllPermissions(AllPermissionsReq) returns (AllPermissionsResponse); + + // ### Search permission sets + // Returns all permission set records that match the given search criteria. + // If multiple search params are given and `filter_or` is FALSE or not specified, + // search params are combined in a logical AND operation. + // Only rows that match *all* search param criteria will be returned. + // + // If `filter_or` is TRUE, multiple search params are combined in a logical OR operation. + // Results will include rows that match **any** of the search criteria. + // + // String search params use case-insensitive matching. + // String search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions. + // example="dan%" will match "danger" and "Danzig" but not "David" + // example="D_m%" will match "Damage" and "dump" + // + // Integer search params can accept a single value or a comma separated list of values. The multiple + // values will be combined under a logical OR operation - results will match at least one of + // the given values. + // + // Most search params can accept "IS NULL" and "NOT NULL" as special expressions to match + // or exclude (respectively) rows where the column is null. + // + // Boolean search params accept only "true" and "false" as values. + // + // + rpc SearchPermissionSets(SearchPermissionSetsReq) returns (SearchPermissionSetsResponse); + + // ### Get information about the permission set with a specific id. + // + rpc PermissionSet(PermissionSetReq) returns (PermissionSetResponse); + + // ### Update information about the permission set with a specific id. + // + rpc UpdatePermissionSet(UpdatePermissionSetReq) returns (UpdatePermissionSetResponse); + + // ### Delete the permission set with a specific id. + // + rpc DeletePermissionSet(DeletePermissionSetReq) returns (DeletePermissionSetResponse); + + // ### Get information about all permission sets. + // + rpc AllPermissionSets(AllPermissionSetsReq) returns (AllPermissionSetsResponse); + + // ### Create a permission set with the specified information. Permission sets are used by Roles. + // + rpc CreatePermissionSet(CreatePermissionSetReq) returns (CreatePermissionSetResponse); + + // ### Get information about all roles. + // + rpc AllRoles(AllRolesReq) returns (AllRolesResponse); + + // ### Create a role with the specified information. + // + rpc CreateRole(CreateRoleReq) returns (CreateRoleResponse); + + // ### Search roles + // + // Returns all role records that match the given search criteria. + // + // If multiple search params are given and `filter_or` is FALSE or not specified, + // search params are combined in a logical AND operation. + // Only rows that match *all* search param criteria will be returned. + // + // If `filter_or` is TRUE, multiple search params are combined in a logical OR operation. + // Results will include rows that match **any** of the search criteria. + // + // String search params use case-insensitive matching. + // String search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions. + // example="dan%" will match "danger" and "Danzig" but not "David" + // example="D_m%" will match "Damage" and "dump" + // + // Integer search params can accept a single value or a comma separated list of values. The multiple + // values will be combined under a logical OR operation - results will match at least one of + // the given values. + // + // Most search params can accept "IS NULL" and "NOT NULL" as special expressions to match + // or exclude (respectively) rows where the column is null. + // + // Boolean search params accept only "true" and "false" as values. + // + // + rpc SearchRoles(SearchRolesReq) returns (SearchRolesResponse); + + // ### Search roles include user count + // + // Returns all role records that match the given search criteria, and attaches + // associated user counts. + // + // If multiple search params are given and `filter_or` is FALSE or not specified, + // search params are combined in a logical AND operation. + // Only rows that match *all* search param criteria will be returned. + // + // If `filter_or` is TRUE, multiple search params are combined in a logical OR operation. + // Results will include rows that match **any** of the search criteria. + // + // String search params use case-insensitive matching. + // String search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions. + // example="dan%" will match "danger" and "Danzig" but not "David" + // example="D_m%" will match "Damage" and "dump" + // + // Integer search params can accept a single value or a comma separated list of values. The multiple + // values will be combined under a logical OR operation - results will match at least one of + // the given values. + // + // Most search params can accept "IS NULL" and "NOT NULL" as special expressions to match + // or exclude (respectively) rows where the column is null. + // + // Boolean search params accept only "true" and "false" as values. + // + // + rpc SearchRolesWithUserCount(SearchRolesWithUserCountReq) returns (SearchRolesWithUserCountResponse); + + // ### Get information about the role with a specific id. + // + rpc Role(RoleReq) returns (RoleResponse); + + // ### Update information about the role with a specific id. + // + rpc UpdateRole(UpdateRoleReq) returns (UpdateRoleResponse); + + // ### Delete the role with a specific id. + // + rpc DeleteRole(DeleteRoleReq) returns (DeleteRoleResponse); + + // ### Get information about all the groups with the role that has a specific id. + // + rpc RoleGroups(RoleGroupsReq) returns (RoleGroupsResponse); + + // ### Set all groups for a role, removing all existing group associations from that role. + // + rpc SetRoleGroups(SetRoleGroupsReq) returns (SetRoleGroupsResponse); + + // ### Get information about all the users with the role that has a specific id. + // + rpc RoleUsers(RoleUsersReq) returns (RoleUsersResponse); + + // ### Set all the users of the role with a specific id. + // + rpc SetRoleUsers(SetRoleUsersReq) returns (SetRoleUsersResponse); + + + + // ScheduledPlan: Manage Scheduled Plans + + // ### Get Scheduled Plans for a Space + // + // Returns scheduled plans owned by the caller for a given space id. + // + rpc ScheduledPlansForSpace(ScheduledPlansForSpaceReq) returns (ScheduledPlansForSpaceResponse); + + // ### Get Information About a Scheduled Plan + // + // Admins can fetch information about other users' Scheduled Plans. + // + rpc ScheduledPlan(ScheduledPlanReq) returns (ScheduledPlanResponse); + + // ### Update a Scheduled Plan + // + // Admins can update other users' Scheduled Plans. + // + // Note: Any scheduled plan destinations specified in an update will **replace** all scheduled plan destinations + // currently defined for the scheduled plan. + // + // For Example: If a scheduled plan has destinations A, B, and C, and you call update on this scheduled plan + // specifying only B in the destinations, then destinations A and C will be deleted by the update. + // + // Updating a scheduled plan to assign null or an empty array to the scheduled_plan_destinations property is an error, as a scheduled plan must always have at least one destination. + // + // If you omit the scheduled_plan_destinations property from the object passed to update, then the destinations + // defined on the original scheduled plan will remain unchanged. + // + // #### Email Permissions: + // + // For details about permissions required to schedule delivery to email and the safeguards + // Looker offers to protect against sending to unauthorized email destinations, see [Email Domain Whitelist for Scheduled Looks](https://docs.looker.com/r/api/embed-permissions). + // + // + // #### Scheduled Plan Destination Formats + // + // Scheduled plan destinations must specify the data format to produce and send to the destination. + // + // Formats: + // + // | format | Description + // | :-----------: | :--- | + // | json | A JSON object containing a `data` property which contains an array of JSON objects, one per row. No metadata. + // | json_detail | Row data plus metadata describing the fields, pivots, table calcs, and other aspects of the query + // | inline_json | Same as the JSON format, except that the `data` property is a string containing JSON-escaped row data. Additional properties describe the data operation. This format is primarily used to send data to web hooks so that the web hook doesn't have to re-encode the JSON row data in order to pass it on to its ultimate destination. + // | csv | Comma separated values with a header + // | txt | Tab separated values with a header + // | html | Simple html + // | xlsx | MS Excel spreadsheet + // | wysiwyg_pdf | Dashboard rendered in a tiled layout to produce a PDF document + // | assembled_pdf | Dashboard rendered in a single column layout to produce a PDF document + // | wysiwyg_png | Dashboard rendered in a tiled layout to produce a PNG image + // || + // + // Valid formats vary by destination type and source object. `wysiwyg_pdf` is only valid for dashboards, for example. + // + // + // + rpc UpdateScheduledPlan(UpdateScheduledPlanReq) returns (UpdateScheduledPlanResponse); + + // ### Delete a Scheduled Plan + // + // Normal users can only delete their own scheduled plans. + // Admins can delete other users' scheduled plans. + // This delete cannot be undone. + // + rpc DeleteScheduledPlan(DeleteScheduledPlanReq) returns (DeleteScheduledPlanResponse); + + // ### List All Scheduled Plans + // + // Returns all scheduled plans which belong to the caller or given user. + // + // If no user_id is provided, this function returns the scheduled plans owned by the caller. + // + // + // To list all schedules for all users, pass `all_users=true`. + // + // + // The caller must have `see_schedules` permission to see other users' scheduled plans. + // + // + // + rpc AllScheduledPlans(AllScheduledPlansReq) returns (AllScheduledPlansResponse); + + // ### Create a Scheduled Plan + // + // Create a scheduled plan to render a Look or Dashboard on a recurring schedule. + // + // To create a scheduled plan, you MUST provide values for the following fields: + // `name` + // and + // `look_id`, `dashboard_id`, `lookml_dashboard_id`, or `query_id` + // and + // `cron_tab` or `datagroup` + // and + // at least one scheduled_plan_destination + // + // A scheduled plan MUST have at least one scheduled_plan_destination defined. + // + // When `look_id` is set, `require_no_results`, `require_results`, and `require_change` are all required. + // + // If `create_scheduled_plan` fails with a 422 error, be sure to look at the error messages in the response which will explain exactly what fields are missing or values that are incompatible. + // + // The queries that provide the data for the look or dashboard are run in the context of user account that owns the scheduled plan. + // + // When `run_as_recipient` is `false` or not specified, the queries that provide the data for the + // look or dashboard are run in the context of user account that owns the scheduled plan. + // + // When `run_as_recipient` is `true` and all the email recipients are Looker user accounts, the + // queries are run in the context of each recipient, so different recipients may see different + // data from the same scheduled render of a look or dashboard. For more details, see [Run As Recipient](https://looker.com/docs/r/admin/run-as-recipient). + // + // Admins can create and modify scheduled plans on behalf of other users by specifying a user id. + // Non-admin users may not create or modify scheduled plans by or for other users. + // + // #### Email Permissions: + // + // For details about permissions required to schedule delivery to email and the safeguards + // Looker offers to protect against sending to unauthorized email destinations, see [Email Domain Whitelist for Scheduled Looks](https://docs.looker.com/r/api/embed-permissions). + // + // + // #### Scheduled Plan Destination Formats + // + // Scheduled plan destinations must specify the data format to produce and send to the destination. + // + // Formats: + // + // | format | Description + // | :-----------: | :--- | + // | json | A JSON object containing a `data` property which contains an array of JSON objects, one per row. No metadata. + // | json_detail | Row data plus metadata describing the fields, pivots, table calcs, and other aspects of the query + // | inline_json | Same as the JSON format, except that the `data` property is a string containing JSON-escaped row data. Additional properties describe the data operation. This format is primarily used to send data to web hooks so that the web hook doesn't have to re-encode the JSON row data in order to pass it on to its ultimate destination. + // | csv | Comma separated values with a header + // | txt | Tab separated values with a header + // | html | Simple html + // | xlsx | MS Excel spreadsheet + // | wysiwyg_pdf | Dashboard rendered in a tiled layout to produce a PDF document + // | assembled_pdf | Dashboard rendered in a single column layout to produce a PDF document + // | wysiwyg_png | Dashboard rendered in a tiled layout to produce a PNG image + // || + // + // Valid formats vary by destination type and source object. `wysiwyg_pdf` is only valid for dashboards, for example. + // + // + // + rpc CreateScheduledPlan(CreateScheduledPlanReq) returns (CreateScheduledPlanResponse); + + // ### Run a Scheduled Plan Immediately + // + // Create a scheduled plan that runs only once, and immediately. + // + // This can be useful for testing a Scheduled Plan before committing to a production schedule. + // + // Admins can create scheduled plans on behalf of other users by specifying a user id. + // + // This API is rate limited to prevent it from being used for relay spam or DoS attacks + // + // #### Email Permissions: + // + // For details about permissions required to schedule delivery to email and the safeguards + // Looker offers to protect against sending to unauthorized email destinations, see [Email Domain Whitelist for Scheduled Looks](https://docs.looker.com/r/api/embed-permissions). + // + // + // #### Scheduled Plan Destination Formats + // + // Scheduled plan destinations must specify the data format to produce and send to the destination. + // + // Formats: + // + // | format | Description + // | :-----------: | :--- | + // | json | A JSON object containing a `data` property which contains an array of JSON objects, one per row. No metadata. + // | json_detail | Row data plus metadata describing the fields, pivots, table calcs, and other aspects of the query + // | inline_json | Same as the JSON format, except that the `data` property is a string containing JSON-escaped row data. Additional properties describe the data operation. This format is primarily used to send data to web hooks so that the web hook doesn't have to re-encode the JSON row data in order to pass it on to its ultimate destination. + // | csv | Comma separated values with a header + // | txt | Tab separated values with a header + // | html | Simple html + // | xlsx | MS Excel spreadsheet + // | wysiwyg_pdf | Dashboard rendered in a tiled layout to produce a PDF document + // | assembled_pdf | Dashboard rendered in a single column layout to produce a PDF document + // | wysiwyg_png | Dashboard rendered in a tiled layout to produce a PNG image + // || + // + // Valid formats vary by destination type and source object. `wysiwyg_pdf` is only valid for dashboards, for example. + // + // + // + rpc ScheduledPlanRunOnce(ScheduledPlanRunOnceReq) returns (ScheduledPlanRunOnceResponse); + + // ### Get Scheduled Plans for a Look + // + // Returns all scheduled plans for a look which belong to the caller or given user. + // + // If no user_id is provided, this function returns the scheduled plans owned by the caller. + // + // + // To list all schedules for all users, pass `all_users=true`. + // + // + // The caller must have `see_schedules` permission to see other users' scheduled plans. + // + // + // + rpc ScheduledPlansForLook(ScheduledPlansForLookReq) returns (ScheduledPlansForLookResponse); + + // ### Get Scheduled Plans for a Dashboard + // + // Returns all scheduled plans for a dashboard which belong to the caller or given user. + // + // If no user_id is provided, this function returns the scheduled plans owned by the caller. + // + // + // To list all schedules for all users, pass `all_users=true`. + // + // + // The caller must have `see_schedules` permission to see other users' scheduled plans. + // + // + // + rpc ScheduledPlansForDashboard(ScheduledPlansForDashboardReq) returns (ScheduledPlansForDashboardResponse); + + // ### Get Scheduled Plans for a LookML Dashboard + // + // Returns all scheduled plans for a LookML Dashboard which belong to the caller or given user. + // + // If no user_id is provided, this function returns the scheduled plans owned by the caller. + // + // + // To list all schedules for all users, pass `all_users=true`. + // + // + // The caller must have `see_schedules` permission to see other users' scheduled plans. + // + // + // + rpc ScheduledPlansForLookmlDashboard(ScheduledPlansForLookmlDashboardReq) returns (ScheduledPlansForLookmlDashboardResponse); + + // ### Run a Scheduled Plan By Id Immediately + // This function creates a run-once schedule plan based on an existing scheduled plan, + // applies modifications (if any) to the new scheduled plan, and runs the new schedule plan immediately. + // This can be useful for testing modifications to an existing scheduled plan before committing to a production schedule. + // + // This function internally performs the following operations: + // + // 1. Copies the properties of the existing scheduled plan into a new scheduled plan + // 2. Copies any properties passed in the JSON body of this request into the new scheduled plan (replacing the original values) + // 3. Creates the new scheduled plan + // 4. Runs the new scheduled plan + // + // The original scheduled plan is not modified by this operation. + // Admins can create, modify, and run scheduled plans on behalf of other users by specifying a user id. + // Non-admins can only create, modify, and run their own scheduled plans. + // + // #### Email Permissions: + // + // For details about permissions required to schedule delivery to email and the safeguards + // Looker offers to protect against sending to unauthorized email destinations, see [Email Domain Whitelist for Scheduled Looks](https://docs.looker.com/r/api/embed-permissions). + // + // + // #### Scheduled Plan Destination Formats + // + // Scheduled plan destinations must specify the data format to produce and send to the destination. + // + // Formats: + // + // | format | Description + // | :-----------: | :--- | + // | json | A JSON object containing a `data` property which contains an array of JSON objects, one per row. No metadata. + // | json_detail | Row data plus metadata describing the fields, pivots, table calcs, and other aspects of the query + // | inline_json | Same as the JSON format, except that the `data` property is a string containing JSON-escaped row data. Additional properties describe the data operation. This format is primarily used to send data to web hooks so that the web hook doesn't have to re-encode the JSON row data in order to pass it on to its ultimate destination. + // | csv | Comma separated values with a header + // | txt | Tab separated values with a header + // | html | Simple html + // | xlsx | MS Excel spreadsheet + // | wysiwyg_pdf | Dashboard rendered in a tiled layout to produce a PDF document + // | assembled_pdf | Dashboard rendered in a single column layout to produce a PDF document + // | wysiwyg_png | Dashboard rendered in a tiled layout to produce a PNG image + // || + // + // Valid formats vary by destination type and source object. `wysiwyg_pdf` is only valid for dashboards, for example. + // + // + // + // This API is rate limited to prevent it from being used for relay spam or DoS attacks + // + // + rpc ScheduledPlanRunOnceById(ScheduledPlanRunOnceByIdReq) returns (ScheduledPlanRunOnceByIdResponse); + + + + // Session: Session Information + + // ### Get API Session + // + // Returns information about the current API session, such as which workspace is selected for the session. + // + rpc Session(SessionReq) returns (SessionResponse); + + // ### Update API Session + // + // #### API Session Workspace + // + // You can use this endpoint to change the active workspace for the current API session. + // + // Only one workspace can be active in a session. The active workspace can be changed + // any number of times in a session. + // + // The default workspace for API sessions is the "production" workspace. + // + // All Looker APIs that use projects or lookml models (such as running queries) will + // use the version of project and model files defined by this workspace for the lifetime of the + // current API session or until the session workspace is changed again. + // + // An API session has the same lifetime as the access_token used to authenticate API requests. Each successful + // API login generates a new access_token and a new API session. + // + // If your Looker API client application needs to work in a dev workspace across multiple + // API sessions, be sure to select the dev workspace after each login. + // + rpc UpdateSession(UpdateSessionReq) returns (UpdateSessionResponse); + + + + // Theme: Manage Themes + + // ### Get an array of all existing themes + // + // Get a **single theme** by id with [Theme](#!/Theme/theme) + // + // This method returns an array of all existing themes. The active time for the theme is not considered. + // + // **Note**: Custom themes needs to be enabled by Looker. Unless custom themes are enabled, only the automatically generated default theme can be used. Please contact your Account Manager or help.looker.com to update your license for this feature. + // + // + rpc AllThemes(AllThemesReq) returns (AllThemesResponse); + + // ### Create a theme + // + // Creates a new theme object, returning the theme details, including the created id. + // + // If `settings` are not specified, the default theme settings will be copied into the new theme. + // + // The theme `name` can only contain alphanumeric characters or underscores. Theme names should not contain any confidential information, such as customer names. + // + // **Update** an existing theme with [Update Theme](#!/Theme/update_theme) + // + // **Permanently delete** an existing theme with [Delete Theme](#!/Theme/delete_theme) + // + // For more information, see [Creating and Applying Themes](https://looker.com/docs/r/admin/themes). + // + // **Note**: Custom themes needs to be enabled by Looker. Unless custom themes are enabled, only the automatically generated default theme can be used. Please contact your Account Manager or help.looker.com to update your license for this feature. + // + // + rpc CreateTheme(CreateThemeReq) returns (CreateThemeResponse); + + // ### Search all themes for matching criteria. + // + // Returns an **array of theme objects** that match the specified search criteria. + // + // | Search Parameters | Description + // | :-------------------: | :------ | + // | `begin_at` only | Find themes active at or after `begin_at` + // | `end_at` only | Find themes active at or before `end_at` + // | both set | Find themes with an active inclusive period between `begin_at` and `end_at` + // + // Note: Range matching requires boolean AND logic. + // When using `begin_at` and `end_at` together, do not use `filter_or`=TRUE + // + // If multiple search params are given and `filter_or` is FALSE or not specified, + // search params are combined in a logical AND operation. + // Only rows that match *all* search param criteria will be returned. + // + // If `filter_or` is TRUE, multiple search params are combined in a logical OR operation. + // Results will include rows that match **any** of the search criteria. + // + // String search params use case-insensitive matching. + // String search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions. + // example="dan%" will match "danger" and "Danzig" but not "David" + // example="D_m%" will match "Damage" and "dump" + // + // Integer search params can accept a single value or a comma separated list of values. The multiple + // values will be combined under a logical OR operation - results will match at least one of + // the given values. + // + // Most search params can accept "IS NULL" and "NOT NULL" as special expressions to match + // or exclude (respectively) rows where the column is null. + // + // Boolean search params accept only "true" and "false" as values. + // + // + // Get a **single theme** by id with [Theme](#!/Theme/theme) + // + // **Note**: Custom themes needs to be enabled by Looker. Unless custom themes are enabled, only the automatically generated default theme can be used. Please contact your Account Manager or help.looker.com to update your license for this feature. + // + // + rpc SearchThemes(SearchThemesReq) returns (SearchThemesResponse); + + // ### Get the default theme + // + // Returns the active theme object set as the default. + // + // The **default** theme name can be set in the UI on the Admin|Theme UI page + // + // The optional `ts` parameter can specify a different timestamp than "now." If specified, it returns the default theme at the time indicated. + // + rpc DefaultTheme(DefaultThemeReq) returns (DefaultThemeResponse); + + // ### Set the global default theme by theme name + // + // Only Admin users can call this function. + // + // Only an active theme with no expiration (`end_at` not set) can be assigned as the default theme. As long as a theme has an active record with no expiration, it can be set as the default. + // + // [Create Theme](#!/Theme/create) has detailed information on rules for default and active themes + // + // Returns the new specified default theme object. + // + // **Note**: Custom themes needs to be enabled by Looker. Unless custom themes are enabled, only the automatically generated default theme can be used. Please contact your Account Manager or help.looker.com to update your license for this feature. + // + // + rpc SetDefaultTheme(SetDefaultThemeReq) returns (SetDefaultThemeResponse); + + // ### Get active themes + // + // Returns an array of active themes. + // + // If the `name` parameter is specified, it will return an array with one theme if it's active and found. + // + // The optional `ts` parameter can specify a different timestamp than "now." + // + // **Note**: Custom themes needs to be enabled by Looker. Unless custom themes are enabled, only the automatically generated default theme can be used. Please contact your Account Manager or help.looker.com to update your license for this feature. + // + // + // + rpc ActiveThemes(ActiveThemesReq) returns (ActiveThemesResponse); + + // ### Get the named theme if it's active. Otherwise, return the default theme + // + // The optional `ts` parameter can specify a different timestamp than "now." + // Note: API users with `show` ability can call this function + // + // **Note**: Custom themes needs to be enabled by Looker. Unless custom themes are enabled, only the automatically generated default theme can be used. Please contact your Account Manager or help.looker.com to update your license for this feature. + // + // + rpc ThemeOrDefault(ThemeOrDefaultReq) returns (ThemeOrDefaultResponse); + + // ### Validate a theme with the specified information + // + // Validates all values set for the theme, returning any errors encountered, or 200 OK if valid + // + // See [Create Theme](#!/Theme/create_theme) for constraints + // + // **Note**: Custom themes needs to be enabled by Looker. Unless custom themes are enabled, only the automatically generated default theme can be used. Please contact your Account Manager or help.looker.com to update your license for this feature. + // + // + rpc ValidateTheme(ValidateThemeReq) returns (ValidateThemeResponse); + + // ### Get a theme by ID + // + // Use this to retrieve a specific theme, whether or not it's currently active. + // + // **Note**: Custom themes needs to be enabled by Looker. Unless custom themes are enabled, only the automatically generated default theme can be used. Please contact your Account Manager or help.looker.com to update your license for this feature. + // + // + rpc Theme(ThemeReq) returns (ThemeResponse); + + // ### Update the theme by id. + // + // **Note**: Custom themes needs to be enabled by Looker. Unless custom themes are enabled, only the automatically generated default theme can be used. Please contact your Account Manager or help.looker.com to update your license for this feature. + // + // + rpc UpdateTheme(UpdateThemeReq) returns (UpdateThemeResponse); + + // ### Delete a specific theme by id + // + // This operation permanently deletes the identified theme from the database. + // + // Because multiple themes can have the same name (with different activation time spans) themes can only be deleted by ID. + // + // All IDs associated with a theme name can be retrieved by searching for the theme name with [Theme Search](#!/Theme/search). + // + // **Note**: Custom themes needs to be enabled by Looker. Unless custom themes are enabled, only the automatically generated default theme can be used. Please contact your Account Manager or help.looker.com to update your license for this feature. + // + // + rpc DeleteTheme(DeleteThemeReq) returns (DeleteThemeResponse); + + + + // User: Manage Users + + // ### Search email credentials + // + // Returns all credentials_email records that match the given search criteria. + // + // If multiple search params are given and `filter_or` is FALSE or not specified, + // search params are combined in a logical AND operation. + // Only rows that match *all* search param criteria will be returned. + // + // If `filter_or` is TRUE, multiple search params are combined in a logical OR operation. + // Results will include rows that match **any** of the search criteria. + // + // String search params use case-insensitive matching. + // String search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions. + // example="dan%" will match "danger" and "Danzig" but not "David" + // example="D_m%" will match "Damage" and "dump" + // + // Integer search params can accept a single value or a comma separated list of values. The multiple + // values will be combined under a logical OR operation - results will match at least one of + // the given values. + // + // Most search params can accept "IS NULL" and "NOT NULL" as special expressions to match + // or exclude (respectively) rows where the column is null. + // + // Boolean search params accept only "true" and "false" as values. + // + // + rpc SearchCredentialsEmail(SearchCredentialsEmailReq) returns (SearchCredentialsEmailResponse); + + // ### Get information about the current user; i.e. the user account currently calling the API. + // + rpc Me(MeReq) returns (MeResponse); + + // ### Get information about all users. + // + rpc AllUsers(AllUsersReq) returns (AllUsersResponse); + + // ### Create a user with the specified information. + // + rpc CreateUser(CreateUserReq) returns (CreateUserResponse); + + // ### Search users + // + // Returns all* user records that match the given search criteria. + // + // If multiple search params are given and `filter_or` is FALSE or not specified, + // search params are combined in a logical AND operation. + // Only rows that match *all* search param criteria will be returned. + // + // If `filter_or` is TRUE, multiple search params are combined in a logical OR operation. + // Results will include rows that match **any** of the search criteria. + // + // String search params use case-insensitive matching. + // String search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions. + // example="dan%" will match "danger" and "Danzig" but not "David" + // example="D_m%" will match "Damage" and "dump" + // + // Integer search params can accept a single value or a comma separated list of values. The multiple + // values will be combined under a logical OR operation - results will match at least one of + // the given values. + // + // Most search params can accept "IS NULL" and "NOT NULL" as special expressions to match + // or exclude (respectively) rows where the column is null. + // + // Boolean search params accept only "true" and "false" as values. + // + // + // (*) Results are always filtered to the level of information the caller is permitted to view. + // Looker admins can see all user details; normal users in an open system can see + // names of other users but no details; normal users in a closed system can only see + // names of other users who are members of the same group as the user. + // + // + rpc SearchUsers(SearchUsersReq) returns (SearchUsersResponse); + + // ### Search for user accounts by name + // + // Returns all user accounts where `first_name` OR `last_name` OR `email` field values match a pattern. + // The pattern can contain `%` and `_` wildcards as in SQL LIKE expressions. + // + // Any additional search params will be combined into a logical AND expression. + // + rpc SearchUsersNames(SearchUsersNamesReq) returns (SearchUsersNamesResponse); + + // ### Get information about the user with a specific id. + // + // If the caller is an admin or the caller is the user being specified, then full user information will + // be returned. Otherwise, a minimal 'public' variant of the user information will be returned. This contains + // The user name and avatar url, but no sensitive information. + // + rpc User(UserReq) returns (UserResponse); + + // ### Update information about the user with a specific id. + // + rpc UpdateUser(UpdateUserReq) returns (UpdateUserResponse); + + // ### Delete the user with a specific id. + // + // **DANGER** this will delete the user and all looks and other information owned by the user. + // + rpc DeleteUser(DeleteUserReq) returns (DeleteUserResponse); + + // ### Get information about the user with a credential of given type with specific id. + // + // This is used to do things like find users by their embed external_user_id. Or, find the user with + // a given api3 client_id, etc. The 'credential_type' matches the 'type' name of the various credential + // types. It must be one of the values listed in the table below. The 'credential_id' is your unique Id + // for the user and is specific to each type of credential. + // + // An example using the Ruby sdk might look like: + // + // `sdk.user_for_credential('embed', 'customer-4959425')` + // + // This table shows the supported 'Credential Type' strings. The right column is for reference; it shows + // which field in the given credential type is actually searched when finding a user with the supplied + // 'credential_id'. + // + // | Credential Types | Id Field Matched | + // | ---------------- | ---------------- | + // | email | email | + // | google | google_user_id | + // | saml | saml_user_id | + // | oidc | oidc_user_id | + // | ldap | ldap_id | + // | api | token | + // | api3 | client_id | + // | embed | external_user_id | + // | looker_openid | email | + // + // **NOTE**: The 'api' credential type was only used with the legacy Looker query API and is no longer supported. The credential type for API you are currently looking at is 'api3'. + // + // + rpc UserForCredential(UserForCredentialReq) returns (UserForCredentialResponse); + + // ### Email/password login information for the specified user. + rpc UserCredentialsEmail(UserCredentialsEmailReq) returns (UserCredentialsEmailResponse); + + // ### Email/password login information for the specified user. + rpc CreateUserCredentialsEmail(CreateUserCredentialsEmailReq) returns (CreateUserCredentialsEmailResponse); + + // ### Email/password login information for the specified user. + rpc UpdateUserCredentialsEmail(UpdateUserCredentialsEmailReq) returns (UpdateUserCredentialsEmailResponse); + + // ### Email/password login information for the specified user. + rpc DeleteUserCredentialsEmail(DeleteUserCredentialsEmailReq) returns (DeleteUserCredentialsEmailResponse); + + // ### Two-factor login information for the specified user. + rpc UserCredentialsTotp(UserCredentialsTotpReq) returns (UserCredentialsTotpResponse); + + // ### Two-factor login information for the specified user. + rpc CreateUserCredentialsTotp(CreateUserCredentialsTotpReq) returns (CreateUserCredentialsTotpResponse); + + // ### Two-factor login information for the specified user. + rpc DeleteUserCredentialsTotp(DeleteUserCredentialsTotpReq) returns (DeleteUserCredentialsTotpResponse); + + // ### LDAP login information for the specified user. + rpc UserCredentialsLdap(UserCredentialsLdapReq) returns (UserCredentialsLdapResponse); + + // ### LDAP login information for the specified user. + rpc DeleteUserCredentialsLdap(DeleteUserCredentialsLdapReq) returns (DeleteUserCredentialsLdapResponse); + + // ### Google authentication login information for the specified user. + rpc UserCredentialsGoogle(UserCredentialsGoogleReq) returns (UserCredentialsGoogleResponse); + + // ### Google authentication login information for the specified user. + rpc DeleteUserCredentialsGoogle(DeleteUserCredentialsGoogleReq) returns (DeleteUserCredentialsGoogleResponse); + + // ### Saml authentication login information for the specified user. + rpc UserCredentialsSaml(UserCredentialsSamlReq) returns (UserCredentialsSamlResponse); + + // ### Saml authentication login information for the specified user. + rpc DeleteUserCredentialsSaml(DeleteUserCredentialsSamlReq) returns (DeleteUserCredentialsSamlResponse); + + // ### OpenID Connect (OIDC) authentication login information for the specified user. + rpc UserCredentialsOidc(UserCredentialsOidcReq) returns (UserCredentialsOidcResponse); + + // ### OpenID Connect (OIDC) authentication login information for the specified user. + rpc DeleteUserCredentialsOidc(DeleteUserCredentialsOidcReq) returns (DeleteUserCredentialsOidcResponse); + + // ### API 3 login information for the specified user. This is for the newer API keys that can be added for any user. + rpc UserCredentialsApi3(UserCredentialsApi3Req) returns (UserCredentialsApi3Response); + + // ### API 3 login information for the specified user. This is for the newer API keys that can be added for any user. + rpc DeleteUserCredentialsApi3(DeleteUserCredentialsApi3Req) returns (DeleteUserCredentialsApi3Response); + + // ### API 3 login information for the specified user. This is for the newer API keys that can be added for any user. + rpc AllUserCredentialsApi3s(AllUserCredentialsApi3sReq) returns (AllUserCredentialsApi3sResponse); + + // ### API 3 login information for the specified user. This is for the newer API keys that can be added for any user. + rpc CreateUserCredentialsApi3(CreateUserCredentialsApi3Req) returns (CreateUserCredentialsApi3Response); + + // ### Embed login information for the specified user. + rpc UserCredentialsEmbed(UserCredentialsEmbedReq) returns (UserCredentialsEmbedResponse); + + // ### Embed login information for the specified user. + rpc DeleteUserCredentialsEmbed(DeleteUserCredentialsEmbedReq) returns (DeleteUserCredentialsEmbedResponse); + + // ### Embed login information for the specified user. + rpc AllUserCredentialsEmbeds(AllUserCredentialsEmbedsReq) returns (AllUserCredentialsEmbedsResponse); + + // ### Looker Openid login information for the specified user. Used by Looker Analysts. + rpc UserCredentialsLookerOpenid(UserCredentialsLookerOpenidReq) returns (UserCredentialsLookerOpenidResponse); + + // ### Looker Openid login information for the specified user. Used by Looker Analysts. + rpc DeleteUserCredentialsLookerOpenid(DeleteUserCredentialsLookerOpenidReq) returns (DeleteUserCredentialsLookerOpenidResponse); + + // ### Web login session for the specified user. + rpc UserSession(UserSessionReq) returns (UserSessionResponse); + + // ### Web login session for the specified user. + rpc DeleteUserSession(DeleteUserSessionReq) returns (DeleteUserSessionResponse); + + // ### Web login session for the specified user. + rpc AllUserSessions(AllUserSessionsReq) returns (AllUserSessionsResponse); + + // ### Create a password reset token. + // This will create a cryptographically secure random password reset token for the user. + // If the user already has a password reset token then this invalidates the old token and creates a new one. + // The token is expressed as the 'password_reset_url' of the user's email/password credential object. + // This takes an optional 'expires' param to indicate if the new token should be an expiring token. + // Tokens that expire are typically used for self-service password resets for existing users. + // Invitation emails for new users typically are not set to expire. + // The expire period is always 60 minutes when expires is enabled. + // This method can be called with an empty body. + // + rpc CreateUserCredentialsEmailPasswordReset(CreateUserCredentialsEmailPasswordResetReq) returns (CreateUserCredentialsEmailPasswordResetResponse); + + // ### Get information about roles of a given user + // + rpc UserRoles(UserRolesReq) returns (UserRolesResponse); + + // ### Set roles of the user with a specific id. + // + rpc SetUserRoles(SetUserRolesReq) returns (SetUserRolesResponse); + + // ### Get user attribute values for a given user. + // + // Returns the values of specified user attributes (or all user attributes) for a certain user. + // + // A value for each user attribute is searched for in the following locations, in this order: + // + // 1. in the user's account information + // 1. in groups that the user is a member of + // 1. the default value of the user attribute + // + // If more than one group has a value defined for a user attribute, the group with the lowest rank wins. + // + // The response will only include user attributes for which values were found. Use `include_unset=true` to include + // empty records for user attributes with no value. + // + // The value of all hidden user attributes will be blank. + // + rpc UserAttributeUserValues(UserAttributeUserValuesReq) returns (UserAttributeUserValuesResponse); + + // ### Store a custom value for a user attribute in a user's account settings. + // + // Per-user user attribute values take precedence over group or default values. + // + rpc SetUserAttributeUserValue(SetUserAttributeUserValueReq) returns (SetUserAttributeUserValueResponse); + + // ### Delete a user attribute value from a user's account settings. + // + // After the user attribute value is deleted from the user's account settings, subsequent requests + // for the user attribute value for this user will draw from the user's groups or the default + // value of the user attribute. See [Get User Attribute Values](#!/User/user_attribute_user_values) for more + // information about how user attribute values are resolved. + // + rpc DeleteUserAttributeUserValue(DeleteUserAttributeUserValueReq) returns (DeleteUserAttributeUserValueResponse); + + // ### Send a password reset token. + // This will send a password reset email to the user. If a password reset token does not already exist + // for this user, it will create one and then send it. + // If the user has not yet set up their account, it will send a setup email to the user. + // The URL sent in the email is expressed as the 'password_reset_url' of the user's email/password credential object. + // Password reset URLs will expire in 60 minutes. + // This method can be called with an empty body. + // + rpc SendUserCredentialsEmailPasswordReset(SendUserCredentialsEmailPasswordResetReq) returns (SendUserCredentialsEmailPasswordResetResponse); + + // ### Change a disabled user's email addresses + // + // Allows the admin to change the email addresses for all the user's + // associated credentials. Will overwrite all associated email addresses with + // the value supplied in the 'email' body param. + // The user's 'is_disabled' status must be true. + // + rpc WipeoutUserEmails(WipeoutUserEmailsReq) returns (WipeoutUserEmailsResponse); + + // Create an embed user from an external user ID + // + rpc CreateEmbedUser(CreateEmbedUserReq) returns (CreateEmbedUserResponse); + + + + // UserAttribute: Manage User Attributes + + // ### Get information about all user attributes. + // + rpc AllUserAttributes(AllUserAttributesReq) returns (AllUserAttributesResponse); + + // ### Create a new user attribute + // + // Permission information for a user attribute is conveyed through the `can` and `user_can_edit` fields. + // The `user_can_edit` field indicates whether an attribute is user-editable _anywhere_ in the application. + // The `can` field gives more granular access information, with the `set_value` child field indicating whether + // an attribute's value can be set by [Setting the User Attribute User Value](#!/User/set_user_attribute_user_value). + // + // Note: `name` and `label` fields must be unique across all user attributes in the Looker instance. + // Attempting to create a new user attribute with a name or label that duplicates an existing + // user attribute will fail with a 422 error. + // + rpc CreateUserAttribute(CreateUserAttributeReq) returns (CreateUserAttributeResponse); + + // ### Get information about a user attribute. + // + rpc UserAttribute(UserAttributeReq) returns (UserAttributeResponse); + + // ### Update a user attribute definition. + // + rpc UpdateUserAttribute(UpdateUserAttributeReq) returns (UpdateUserAttributeResponse); + + // ### Delete a user attribute (admin only). + // + rpc DeleteUserAttribute(DeleteUserAttributeReq) returns (DeleteUserAttributeResponse); + + // ### Returns all values of a user attribute defined by user groups, in precedence order. + // + // A user may be a member of multiple groups which define different values for a given user attribute. + // The order of group-values in the response determines precedence for selecting which group-value applies + // to a given user. For more information, see [Set User Attribute Group Values](#!/UserAttribute/set_user_attribute_group_values). + // + // Results will only include groups that the caller's user account has permission to see. + // + rpc AllUserAttributeGroupValues(AllUserAttributeGroupValuesReq) returns (AllUserAttributeGroupValuesResponse); + + // ### Define values for a user attribute across a set of groups, in priority order. + // + // This function defines all values for a user attribute defined by user groups. This is a global setting, potentially affecting + // all users in the system. This function replaces any existing group value definitions for the indicated user attribute. + // + // The value of a user attribute for a given user is determined by searching the following locations, in this order: + // + // 1. the user's account settings + // 2. the groups that the user is a member of + // 3. the default value of the user attribute, if any + // + // The user may be a member of multiple groups which define different values for that user attribute. The order of items in the group_values parameter + // determines which group takes priority for that user. Lowest array index wins. + // + // An alternate method to indicate the selection precedence of group-values is to assign numbers to the 'rank' property of each + // group-value object in the array. Lowest 'rank' value wins. If you use this technique, you must assign a + // rank value to every group-value object in the array. + // + // To set a user attribute value for a single user, see [Set User Attribute User Value](#!/User/set_user_attribute_user_value). + // To set a user attribute value for all members of a group, see [Set User Attribute Group Value](#!/Group/update_user_attribute_group_value). + // + rpc SetUserAttributeGroupValues(SetUserAttributeGroupValuesReq) returns (SetUserAttributeGroupValuesResponse); + + + + // Workspace: Manage Workspaces + + // ### Get All Workspaces + // + // Returns all workspaces available to the calling user. + // + rpc AllWorkspaces(AllWorkspacesReq) returns (AllWorkspacesResponse); + + // ### Get A Workspace + // + // Returns information about a workspace such as the git status and selected branches + // of all projects available to the caller's user account. + // + // A workspace defines which versions of project files will be used to evaluate expressions + // and operations that use model definitions - operations such as running queries or rendering dashboards. + // Each project has its own git repository, and each project in a workspace may be configured to reference + // particular branch or revision within their respective repositories. + // + // There are two predefined workspaces available: "production" and "dev". + // + // The production workspace is shared across all Looker users. Models in the production workspace are read-only. + // Changing files in production is accomplished by modifying files in a git branch and using Pull Requests + // to merge the changes from the dev branch into the production branch, and then telling + // Looker to sync with production. + // + // The dev workspace is local to each Looker user. Changes made to project/model files in the dev workspace only affect + // that user, and only when the dev workspace is selected as the active workspace for the API session. + // (See set_session_workspace()). + // + // The dev workspace is NOT unique to an API session. Two applications accessing the Looker API using + // the same user account will see the same files in the dev workspace. To avoid collisions between + // API clients it's best to have each client login with API3 credentials for a different user account. + // + // Changes made to files in a dev workspace are persistent across API sessions. It's a good + // idea to commit any changes you've made to the git repository, but not strictly required. Your modified files + // reside in a special user-specific directory on the Looker server and will still be there when you login in again + // later and use update_session(workspace_id: "dev") to select the dev workspace for the new API session. + // + rpc Workspace(WorkspaceReq) returns (WorkspaceResponse); + + +} \ No newline at end of file diff --git a/proto/grpc_proxy/src/main/proto/sdk/models.proto b/proto/grpc_proxy/src/main/proto/sdk/models.proto new file mode 100644 index 000000000..1105c7db2 --- /dev/null +++ b/proto/grpc_proxy/src/main/proto/sdk/models.proto @@ -0,0 +1,11807 @@ +// MIT License +// +// Copyright (c) 2021 Looker Data Sciences, Inc. +// +// Permission is hereby granted, free of charge, to any person obtaining a copy +// of this software and associated documentation files (the "Software"), to deal +// in the Software without restriction, including without limitation the rights +// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +// copies of the Software, and to permit persons to whom the Software is +// furnished to do so, subject to the following conditions: +// +// The above copyright notice and this permission notice shall be included in all +// copies or substantial portions of the Software. +// +// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +// SOFTWARE. +// + +// 231 API models: 231 Spec, 0 Request, 0 Write, 0 Enum + + +syntax = "proto3"; + +package looker; + +option java_package = "com.google.looker.grpc.services"; +option java_multiple_files = true; + +import "google/protobuf/any.proto"; +import "google/protobuf/timestamp.proto"; + + +message AccessToken { + // Access Token used for API calls + string access_token = 282617758; + // Type of Token + string token_type = 117094912; + // Number of seconds before the token expires + int64 expires_in = 449388008; + // Refresh token which can be used to obtain a new access token + string refresh_token = 335842954; +} + + +message Alert { + // Filters coming from the dashboard that are applied. Example `[{ "filter_title": "Name", "field_name": "distribution_centers.name", "filter_value": "Los Angeles CA" }]` + repeated AlertAppliedDashboardFilter applied_dashboard_filters = 337449948; + // This property informs the check what kind of comparison we are performing. Only certain condition types are valid for time series alerts. For details, refer to [Setting Alert Conditions](https://docs.looker.com/sharing-and-publishing/creating-alerts#setting_alert_conditions) Valid values are: "EQUAL_TO", "GREATER_THAN", "GREATER_THAN_OR_EQUAL_TO", "LESS_THAN", "LESS_THAN_OR_EQUAL_TO", "INCREASES_BY", "DECREASES_BY", "CHANGES_BY". + ComparisonType comparison_type = 300174680; + // Vixie-Style crontab specification when to run. At minumum, it has to be longer than 15 minute intervals + string cron = 3387314; + // An optional, user-defined title for the alert + string custom_title = 413477701; + // ID of the dashboard element associated with the alert. Refer to [dashboard_element()](#!/Dashboard/DashboardElement) + string dashboard_element_id = 426786746; + // An optional description for the alert. This supplements the title + string description = 375083905; + // Array of destinations to send alerts to. Must be the same type of destination. Example `[{ "destination_type": "EMAIL", "email_address": "test@test.com" }]` + repeated AlertDestination destinations = 166962363; + AlertField field = 95669946; + // Whether or not the user follows this alert. + bool followed = 285909372; + // Whether or not the alert is followable + bool followable = 475402954; + // ID of the alert + string id = 3205; + // Whether or not the alert is disabled + bool is_disabled = 332107427; + // Reason for disabling alert + string disabled_reason = 367441865; + // Whether or not the alert is public + bool is_public = 254583906; + // The type of the investigative content Valid values are: "dashboard". + InvestigativeContentType investigative_content_type = 455691579; + // The ID of the investigative content. For dashboards, this will be the dashboard ID + string investigative_content_id = 329023657; + // The title of the investigative content. + string investigative_content_title = 478413622; + // ID of the LookML dashboard associated with the alert + string lookml_dashboard_id = 351003409; + // ID of the LookML dashboard element associated with the alert + string lookml_link_id = 527451937; + // User id of alert owner + string owner_id = 285850503; + // Alert owner's display name + string owner_display_name = 355700413; + // Value of the alert threshold + double threshold = 425888618; + AlertConditionState time_series_condition_state = 343695767; +} + + +message AlertAppliedDashboardFilter { + // Field Title. Refer to `DashboardFilter.title` in [DashboardFilter](#!/types/DashboardFilter). Example `Name` + string filter_title = 346525976; + // Field Name. Refer to `DashboardFilter.dimension` in [DashboardFilter](#!/types/DashboardFilter). Example `distribution_centers.name` + string field_name = 273359668; + // Field Value. [Filter Expressions](https://docs.looker.com/reference/filter-expressions). Example `Los Angeles CA` + string filter_value = 410804266; + // Human Readable Filter Description. This may be null or auto-generated. Example `is Los Angeles CA` + string filter_description = 484008837; +} + + +message AlertConditionState { + // (Write-Only) The second latest time string the alert has seen. + string previous_time_series_id = 522620129; + // (Write-Only) Latest time string the alert has seen. + string latest_time_series_id = 372921769; +} + + +message AlertDestination { + // Type of destination that the alert will be sent to Valid values are: "EMAIL", "ACTION_HUB". + DestinationType destination_type = 453755050; + // Email address for the 'email' type + string email_address = 347159860; + // Action hub integration id for the 'action_hub' type. [Integration](#!/types/Integration) + string action_hub_integration_id = 309125801; + // Action hub form params json for the 'action_hub' type [IntegrationParam](#!/types/IntegrationParam) + string action_hub_form_params_json = 512583363; +} + + +message AlertField { + // Field's title. Usually auto-generated to reflect field name and its filters + string title = 96607896; + // Field's name. Has the format `.` Refer to [docs](https://docs.looker.com/sharing-and-publishing/creating-alerts) for more details + string name = 3116757; + // (Optional / Advance Use) List of fields filter. This further restricts the alert to certain dashboard element's field values. This can be used on top of dashboard filters `applied_dashboard_filters`. To keep thing simple, it's suggested to just use dashboard filters. Example: `{ 'title': '12 Number on Hand', 'name': 'inventory_items.number_on_hand', 'filter': [{ 'field_name': 'inventory_items.id', 'field_value': 12, 'filter_value': null }] }` + repeated AlertFieldFilter filter = 467202780; +} + + +message AlertFieldFilter { + // Field Name. Has format `.` + string field_name = 273359668; + // Field Value. Depends on the type of field - numeric or string. For [location](https://docs.looker.com/reference/field-reference/dimension-type-reference#location) type, it's a list of floats. Example `[1.0, 56.0]` + google.protobuf.Any field_value = 364569595; + // Filter Value. Usually null except for [location](https://docs.looker.com/reference/field-reference/dimension-type-reference#location) type. It'll be a string of lat,long ie `'1.0,56.0'` + string filter_value = 410804266; +} + + +message AlertPatch { + // New owner ID of the alert + string owner_id = 285850503; + // Set alert enabled or disabled + bool is_disabled = 332107427; + // The reason this alert is disabled + string disabled_reason = 367441865; + // Set alert public or private + bool is_public = 254583906; + // New threshold value + double threshold = 425888618; +} + +// The appropriate horizontal text alignment the values of this field should be displayed in. Valid values are: "left", "right". (Enum defined in LookmlModelExploreField) +enum Align { + _ALIGN_UNSET = 0; + ALIGN_LEFT = 224597503; + ALIGN_RIGHT = 433392158; +} + + +message ApiSession { + // Operations the current user is able to perform on this object + map can = 108816; + // The id of active workspace for this session + string workspace_id = 456431629; + // The id of the actual user in the case when this session represents one user sudo'ing as another + string sudo_user_id = 324097672; +} + + +message ApiVersion { + // Current Looker release version number + string looker_release_version = 512649708; + ApiVersionElement current_version = 274393769; + // Array of versions supported by this Looker instance + repeated ApiVersionElement supported_versions = 294617389; + // API server base url + string api_server_url = 336184798; + // Web server base url + string web_server_url = 336186559; +} + + +message ApiVersionElement { + // Version number as it appears in '/api/xxx/' urls + string version = 529875334; + // Full version number including minor version + string full_version = 426762378; + // Status of this version + string status = 445505145; + // Url for swagger.json for this version + string swagger_url = 350282653; +} + + +message BackupConfiguration { + // Operations the current user is able to perform on this object + map can = 108816; + // Type of backup: looker-s3 or custom-s3 + string type = 3120390; + // Name of bucket for custom-s3 backups + string custom_s3_bucket = 244474693; + // Name of region where the bucket is located + string custom_s3_bucket_region = 511205324; + // (Write-Only) AWS S3 key used for custom-s3 backups + string custom_s3_key = 515118471; + // (Write-Only) AWS S3 secret used for custom-s3 backups + string custom_s3_secret = 387048480; + // Link to get this item + string url = 107439; +} + + +message Board { + // Operations the current user is able to perform on this object + map can = 108816; + // Id of associated content_metadata record + string content_metadata_id = 4111259; + // Date of board creation + google.protobuf.Timestamp created_at = 464589301; + // Date of board deletion + google.protobuf.Timestamp deleted_at = 464526556; + // Description of the board + string description = 375083905; + // Sections of the board + repeated BoardSection board_sections = 456433347; + // Unique Id + string id = 3205; + // ids of the board sections in the order they should be displayed + repeated string section_order = 511535011; + // Title of the board + string title = 96607896; + // Date of last board update + google.protobuf.Timestamp updated_at = 464589803; + // User id of board creator + string user_id = 413336787; + // Whether the board is the primary homepage or not + bool primary_homepage = 4181419; +} + + +message BoardItem { + // Operations the current user is able to perform on this object + map can = 108816; + // Name of user who created the content this item is based on + string content_created_by = 345901194; + // Content favorite id associated with the item this content is based on + string content_favorite_id = 363282886; + // Content metadata id associated with the item this content is based on + string content_metadata_id = 4111259; + // Last time the content that this item is based on was updated + string content_updated_at = 408537127; + // Custom description entered by the user, if present + string custom_description = 517484699; + // Custom title entered by the user, if present + string custom_title = 413477701; + // Custom url entered by the user, if present + string custom_url = 249459585; + // Dashboard to base this item on + string dashboard_id = 311679049; + // The actual description for display + string description = 375083905; + // Number of times content has been favorited, if present + int64 favorite_count = 331565140; + // Associated Board Section + string board_section_id = 473711058; + // Unique Id + string id = 3205; + // The actual image_url for display + string image_url = 384928173; + // The container folder name of the content + string location = 320111754; + // Look to base this item on + string look_id = 413287022; + // LookML Dashboard to base this item on + string lookml_dashboard_id = 351003409; + // An arbitrary integer representing the sort order within the section + int64 order = 108390030; + // The actual title for display + string title = 96607896; + // Relative url for the associated content + string url = 107439; + // Whether the custom description should be used instead of the content description, if the item is associated with content + bool use_custom_description = 532957353; + // Whether the custom title should be used instead of the content title, if the item is associated with content + bool use_custom_title = 533570657; + // Whether the custom url should be used instead of the content url, if the item is associated with content + bool use_custom_url = 441644771; + // Number of times content has been viewed, if present + int64 view_count = 531861914; + // (Write-Only) base64 encoded image data + string custom_image_data_base64 = 427605623; + // Custom image_url entered by the user, if present + string custom_image_url = 497368103; + // Whether the custom image should be used instead of the content image, if the item is associated with content + bool use_custom_image = 398202105; +} + + +message BoardSection { + // Operations the current user is able to perform on this object + map can = 108816; + // Time at which this section was created. + google.protobuf.Timestamp created_at = 464589301; + // Time at which this section was deleted. + google.protobuf.Timestamp deleted_at = 464526556; + // Description of the content found in this section. + string description = 375083905; + // Id reference to parent board + string board_id = 298405268; + // Items in the board section + repeated BoardItem board_items = 391814828; + // Unique Id + string id = 3205; + // ids of the board items in the order they should be displayed + repeated string item_order = 377743503; + // ids of the homepage items the user can see in the order they should be displayed + repeated string visible_item_order = 411259595; + // Name of row + string title = 96607896; + // Time at which this section was last updated. + google.protobuf.Timestamp updated_at = 464589803; +} + +// Field category Valid values are: "parameter", "filter", "measure", "dimension". (Enum defined in LookmlModelExploreField) +enum Category { + _CATEGORY_UNSET = 0; + CATEGORY_PARAMETER = 357851506; + CATEGORY_FILTER = 358303577; + CATEGORY_MEASURE = 382917027; + CATEGORY_DIMENSION = 398805625; +} + + +message ColorCollection { + // Unique Id + string id = 3205; + // Label of color collection + string label = 102846452; + // Array of categorical palette definitions + repeated DiscretePalette categoricalPalettes = 298797876; + // Array of discrete palette definitions + repeated ContinuousPalette sequentialPalettes = 421922795; + // Array of diverging palette definitions + repeated ContinuousPalette divergingPalettes = 402046609; +} + + +message ColorStop { + // CSS color string + string color = 108695523; + // Offset in continuous palette (0 to 100) + int64 offset = 438591449; +} + + +message ColumnSearch { + // Name of schema containing the table + string schema_name = 274052203; + // Name of table containing the column + string table_name = 273589769; + // Name of column + string column_name = 175444588; + // Column data type + string data_type = 278132276; +} + +// This property informs the check what kind of comparison we are performing. Only certain condition types are valid for time series alerts. For details, refer to [Setting Alert Conditions](https://docs.looker.com/sharing-and-publishing/creating-alerts#setting_alert_conditions) Valid values are: "EQUAL_TO", "GREATER_THAN", "GREATER_THAN_OR_EQUAL_TO", "LESS_THAN", "LESS_THAN_OR_EQUAL_TO", "INCREASES_BY", "DECREASES_BY", "CHANGES_BY". (Enum defined in Alert) +enum ComparisonType { + _COMPARISON_TYPE_UNSET = 0; + COMPARISON_TYPE_EQUAL_TO = 505451701; + COMPARISON_TYPE_GREATER_THAN = 437132346; + COMPARISON_TYPE_GREATER_THAN_OR_EQUAL_TO = 304274866; + COMPARISON_TYPE_LESS_THAN = 399542254; + COMPARISON_TYPE_LESS_THAN_OR_EQUAL_TO = 319127449; + COMPARISON_TYPE_INCREASES_BY = 444359942; + COMPARISON_TYPE_DECREASES_BY = 299786180; + COMPARISON_TYPE_CHANGES_BY = 179793186; +} + + +message ConnectionFeatures { + // Name of the dialect for this connection + string dialect_name = 414841156; + // True for cost estimating support + bool cost_estimate = 383492614; + // True for multiple database support + bool multiple_databases = 511040836; + // True for cost estimating support + bool column_search = 306641436; + // True for secondary index support + bool persistent_table_indexes = 491969169; + // True for persistent derived table support + bool persistent_derived_tables = 385679127; + // True for turtles support + bool turtles = 494640445; + // True for percentile support + bool percentile = 280916616; + // True for distinct percentile support + bool distinct_percentile = 349740168; + // True for stable views support + bool stable_views = 299605526; + // True for millisecond support + bool milliseconds = 316682068; + // True for microsecond support + bool microseconds = 315254262; + // True for subtotal support + bool subtotals = 290419819; + // True for geographic location support + bool location = 320111754; + // True for timezone conversion in query support + bool timezone = 516241033; + // True for connection pooling support + bool connection_pooling = 363842273; +} + + +message ContentFavorite { + // Unique Id + string id = 3205; + // User Id which owns this ContentFavorite + string user_id = 413336787; + // Content Metadata Id associated with this ContentFavorite + string content_metadata_id = 4111259; + // Id of a look + string look_id = 413287022; + // Id of a dashboard + string dashboard_id = 311679049; + LookBasic look = 3297857; + DashboardBase dashboard = 79840748; + // Id of a board + string board_id = 298405268; +} + + +message ContentMeta { + // Operations the current user is able to perform on this object + map can = 108816; + // Unique Id + string id = 3205; + // Name or title of underlying content + string name = 3116757; + // Id of Parent Content + string parent_id = 206134256; + // Id of associated dashboard when content_type is "dashboard" + string dashboard_id = 311679049; + // Id of associated look when content_type is "look" + string look_id = 413287022; + // Id of associated folder when content_type is "space" + string folder_id = 271739380; + // Content Type ("dashboard", "look", or "folder") + string content_type = 423262016; + // Whether content inherits its access levels from parent + bool inherits = 386348026; + // Id of Inherited Content + string inheriting_id = 533792493; + // Content Slug + string slug = 3184373; +} + +// WARNING: no writeable properties found for POST, PUT, or PATCH +message ContentMetaGroupUser { + // Operations the current user is able to perform on this object + map can = 108816; + // Unique Id + string id = 3205; + // Id of associated Content Metadata + string content_metadata_id = 4111259; + // Type of permission: "view" or "edit" Valid values are: "view", "edit". + PermissionType permission_type = 336574933; + // ID of associated group + string group_id = 287220091; + // ID of associated user + string user_id = 413336787; +} + + +message ContentValidation { + // A list of content errors + repeated ContentValidatorError content_with_errors = 133537986; + // Duration of content validation in seconds + float computation_time = 483498726; + // The number of looks validated + int64 total_looks_validated = 443398958; + // The number of dashboard elements validated + int64 total_dashboard_elements_validated = 349676325; + // The number of dashboard filters validated + int64 total_dashboard_filters_validated = 371965682; + // The number of scheduled plans validated + int64 total_scheduled_plans_validated = 343740229; + // The number of alerts validated + int64 total_alerts_validated = 358497744; + // The number of explores used across all content validated + int64 total_explores_validated = 465271541; +} + + +message ContentValidationAlert { + // ID of the alert + string id = 3205; + // ID of the LookML dashboard associated with the alert + string lookml_dashboard_id = 351003409; + // ID of the LookML dashboard element associated with the alert + string lookml_link_id = 527451937; + // An optional, user-defined title for the alert + string custom_title = 413477701; +} + + +message ContentValidationDashboard { + // Description + string description = 375083905; + // Unique Id + string id = 3205; + ContentValidationFolder folder = 467441015; + // Dashboard Title + string title = 96607896; + // Relative URL of the dashboard + string url = 107439; +} + + +message ContentValidationDashboardElement { + // Text tile body text + string body_text = 491616469; + // Id of Dashboard + string dashboard_id = 311679049; + // Unique Id + string id = 3205; + // Id Of Look + string look_id = 413287022; + // Note Display + string note_display = 120563243; + // Note State + string note_state = 342745425; + // Note Text + string note_text = 491761577; + // Note Text as Html + string note_text_as_html = 271641721; + // Id Of Query + string query_id = 279007282; + // Text tile subtitle text + string subtitle_text = 507090106; + // Title of dashboard element + string title = 96607896; + // Whether title is hidden + bool title_hidden = 308186743; + // Text tile title + string title_text = 424563514; + // Type + string type = 3120390; + // JSON with all the properties required for rich editor and buttons elements + string rich_content_json = 309924280; +} + + +message ContentValidationDashboardFilter { + // Unique Id + string id = 3205; + // Id of Dashboard + string dashboard_id = 311679049; + // Name of filter + string name = 3116757; + // Title of filter + string title = 96607896; + // Type of filter: one of date, number, string, or field + string type = 3120390; + // Default value of filter + string default_value = 429771091; + // Model of filter (required if type = field) + string model = 102848809; + // Explore of filter (required if type = field) + string explore = 370461451; + // Dimension of filter (required if type = field) + string dimension = 511293107; +} + + +message ContentValidationError { + // Error message + string message = 452371230; + // Name of the field involved in the error + string field_name = 273359668; + // Name of the model involved in the error + string model_name = 275154383; + // Name of the explore involved in the error + string explore_name = 403807127; + // Whether this validation error is removable + bool removable = 523056781; +} + + +message ContentValidationFolder { + // Unique Name + string name = 3116757; + // Unique Id + string id = 3205; +} + + +message ContentValidationLook { + // Unique Id + string id = 3205; + // Look Title + string title = 96607896; + // Short Url + string short_url = 381379717; + ContentValidationFolder folder = 467441015; +} + + +message ContentValidationLookMLDashboard { + // ID of the LookML Dashboard + string id = 3205; + // Title of the LookML Dashboard + string title = 96607896; + // ID of Space + string space_id = 297928564; +} + + +message ContentValidationLookMLDashboardElement { + // Link ID of the LookML Dashboard Element + string lookml_link_id = 527451937; + // Title of the LookML Dashboard Element + string title = 96607896; +} + + +message ContentValidationScheduledPlan { + // Name of this scheduled plan + string name = 3116757; + // Id of a look + string look_id = 413287022; + // Unique Id + string id = 3205; +} + + +message ContentValidatorError { + ContentValidationLook look = 3297857; + ContentValidationDashboard dashboard = 79840748; + ContentValidationDashboardElement dashboard_element = 385801943; + ContentValidationDashboardFilter dashboard_filter = 70871843; + ContentValidationScheduledPlan scheduled_plan = 521105881; + ContentValidationAlert alert = 110625116; + ContentValidationLookMLDashboard lookml_dashboard = 462458611; + ContentValidationLookMLDashboardElement lookml_dashboard_element = 455141060; + // A list of errors found for this piece of content + repeated ContentValidationError errors = 446956773; + // An id unique to this piece of content for this validation run + string id = 3205; +} + + +message ContentView { + // Operations the current user is able to perform on this object + map can = 108816; + // Unique Id + string id = 3205; + // Id of viewed Look + string look_id = 413287022; + // Id of the viewed Dashboard + string dashboard_id = 311679049; + // Name or title of underlying content + string title = 96607896; + // Content metadata id of the Look or Dashboard + string content_metadata_id = 4111259; + // Id of user content was viewed by + string user_id = 413336787; + // Id of group content was viewed by + string group_id = 287220091; + // Number of times piece of content was viewed + int64 view_count = 531861914; + // Number of times piece of content was favorited + int64 favorite_count = 331565140; + // Date the piece of content was last viewed + string last_viewed_at = 510619009; + // Week start date for the view and favorite count during that given week + string start_of_week_date = 513075243; +} + + +message ContinuousPalette { + // Unique identity string + string id = 3205; + // Label for palette + string label = 102846452; + // Type of palette + string type = 3120390; + // Array of ColorStops in the palette + repeated ColorStop stops = 109651889; +} + + +message CostEstimate { + // Cost of SQL statement + int64 cost = 3569811; + // Does the result come from the cache? + bool cache_hit = 272484970; + // Cost measurement size + string cost_unit = 345560829; + // Human-friendly message + string message = 452371230; +} + +// WARNING: no writeable properties found for POST, PUT, or PATCH +message CreateCostEstimate { + // SQL statement to estimate + string sql = 107406; +} + + +message CreateCredentialsApi3 { + // Operations the current user is able to perform on this object + map can = 108816; + // Unique Id + string id = 3205; + // API key client_id + string client_id = 206142577; + // Timestamp for the creation of this credential + string created_at = 464589301; + // Has this credential been disabled? + bool is_disabled = 332107427; + // Short name for the type of this kind of credential + string type = 3120390; + // API key client_secret + string client_secret = 509513545; + // Link to get this item + string url = 107439; +} + + +message CreateDashboardFilter { + // Unique Id + string id = 3205; + // Id of Dashboard + string dashboard_id = 311679049; + // Name of filter + string name = 3116757; + // Title of filter + string title = 96607896; + // Type of filter: one of date, number, string, or field + string type = 3120390; + // Default value of filter + string default_value = 429771091; + // Model of filter (required if type = field) + string model = 102848809; + // Explore of filter (required if type = field) + string explore = 370461451; + // Dimension of filter (required if type = field) + string dimension = 511293107; + // Field information + map field = 95669946; + // Display order of this filter relative to other filters + int64 row = 117914; + // Array of listeners for faceted filters + repeated string listens_to_filters = 493150066; + // Whether the filter allows multiple filter values (deprecated in the latest version of dashboards) + bool allow_multiple_values = 331288581; + // Whether the filter requires a value to run the dashboard + bool required = 497611616; + // The visual configuration for this filter. Used to set up how the UI for this filter should appear. + map ui_config = 413099149; +} + + +message CreateDashboardRenderTask { + // Filter values to apply to the dashboard queries, in URL query format + string dashboard_filters = 470460328; + // Dashboard layout style: single_column or tiled + string dashboard_style = 515014529; +} + + +message CreateEmbedUserRequest { + string external_user_id = 459767842; +} + + +message CreateFolder { + // Unique Name + string name = 3116757; + // Id of Parent. If the parent id is null, this is a root-level entry + string parent_id = 206134256; +} + + +message CreateOAuthApplicationUserStateRequest { + string user_id = 413336787; + string oauth_application_id = 254423058; + string access_token = 282617758; + google.protobuf.Timestamp access_token_expires_at = 475681703; + string refresh_token = 335842954; + google.protobuf.Timestamp refresh_token_expires_at = 494450126; +} + + +message CreateOAuthApplicationUserStateResponse { + // User Id + string user_id = 413336787; + // OAuth Application ID + string oauth_application_id = 254423058; +} + + +message CreateQueryTask { + // Operations the current user is able to perform on this object + map can = 108816; + // Id of query to run + string query_id = 279007282; + // Desired async query result format. Valid values are: "inline_json", "json", "json_detail", "json_fe", "csv", "html", "md", "txt", "xlsx", "gsxml". + ResultFormat result_format = 296518982; + // Source of query task + string source = 327120574; + // Create the task but defer execution + bool deferred = 501523840; + // Id of look associated with query. + string look_id = 413287022; + // Id of dashboard associated with query. + string dashboard_id = 311679049; +} + + +message CredentialsApi3 { + // Operations the current user is able to perform on this object + map can = 108816; + // Unique Id + string id = 3205; + // API key client_id + string client_id = 206142577; + // Timestamp for the creation of this credential + string created_at = 464589301; + // Has this credential been disabled? + bool is_disabled = 332107427; + // Short name for the type of this kind of credential + string type = 3120390; + // Link to get this item + string url = 107439; +} + + +message CredentialsEmail { + // Operations the current user is able to perform on this object + map can = 108816; + // Timestamp for the creation of this credential + string created_at = 464589301; + // EMail address used for user login + string email = 102965020; + // Force the user to change their password upon their next login + bool forced_password_reset_at_next_login = 458418973; + // Has this credential been disabled? + bool is_disabled = 332107427; + // Timestamp for most recent login using credential + string logged_in_at = 411009336; + // Url with one-time use secret token that the user can use to reset password + string password_reset_url = 388986174; + // Short name for the type of this kind of credential + string type = 3120390; + // Link to get this item + string url = 107439; + // Link to get this user + string user_url = 391000882; +} + + +message CredentialsEmailSearch { + // Operations the current user is able to perform on this object + map can = 108816; + // Timestamp for the creation of this credential + string created_at = 464589301; + // EMail address used for user login + string email = 102965020; + // Force the user to change their password upon their next login + bool forced_password_reset_at_next_login = 458418973; + // Has this credential been disabled? + bool is_disabled = 332107427; + // Timestamp for most recent login using credential + string logged_in_at = 411009336; + // Url with one-time use secret token that the user can use to reset password + string password_reset_url = 388986174; + // Short name for the type of this kind of credential + string type = 3120390; + // Link to get this item + string url = 107439; + // Link to get this user + string user_url = 391000882; +} + + +message CredentialsEmbed { + // Operations the current user is able to perform on this object + map can = 108816; + // Timestamp for the creation of this credential + string created_at = 464589301; + // Embedder's id for a group to which this user was added during the most recent login + string external_group_id = 488769304; + // Embedder's unique id for the user + string external_user_id = 459767842; + // Unique Id + string id = 3205; + // Has this credential been disabled? + bool is_disabled = 332107427; + // Timestamp for most recent login using credential + string logged_in_at = 411009336; + // Short name for the type of this kind of credential + string type = 3120390; + // Link to get this item + string url = 107439; +} + + +message CredentialsGoogle { + // Operations the current user is able to perform on this object + map can = 108816; + // Timestamp for the creation of this credential + string created_at = 464589301; + // Google domain + string domain = 522896482; + // EMail address + string email = 102965020; + // Google's Unique ID for this user + string google_user_id = 11617495; + // Has this credential been disabled? + bool is_disabled = 332107427; + // Timestamp for most recent login using credential + string logged_in_at = 411009336; + // Short name for the type of this kind of credential + string type = 3120390; + // Link to get this item + string url = 107439; +} + + +message CredentialsLDAP { + // Operations the current user is able to perform on this object + map can = 108816; + // Timestamp for the creation of this credential + string created_at = 464589301; + // EMail address + string email = 102965020; + // Has this credential been disabled? + bool is_disabled = 332107427; + // LDAP Distinguished name for this user (as-of the last login) + string ldap_dn = 448809928; + // LDAP Unique ID for this user + string ldap_id = 413320812; + // Timestamp for most recent login using credential + string logged_in_at = 411009336; + // Short name for the type of this kind of credential + string type = 3120390; + // Link to get this item + string url = 107439; +} + + +message CredentialsLookerOpenid { + // Operations the current user is able to perform on this object + map can = 108816; + // Timestamp for the creation of this credential + string created_at = 464589301; + // EMail address used for user login + string email = 102965020; + // Has this credential been disabled? + bool is_disabled = 332107427; + // Timestamp for most recent login using credential + string logged_in_at = 411009336; + // IP address of client for most recent login using credential + string logged_in_ip = 296255297; + // Short name for the type of this kind of credential + string type = 3120390; + // Link to get this item + string url = 107439; + // Link to get this user + string user_url = 391000882; +} + + +message CredentialsOIDC { + // Operations the current user is able to perform on this object + map can = 108816; + // Timestamp for the creation of this credential + string created_at = 464589301; + // EMail address + string email = 102965020; + // Has this credential been disabled? + bool is_disabled = 332107427; + // Timestamp for most recent login using credential + string logged_in_at = 411009336; + // OIDC OP's Unique ID for this user + string oidc_user_id = 324008205; + // Short name for the type of this kind of credential + string type = 3120390; + // Link to get this item + string url = 107439; +} + + +message CredentialsSaml { + // Operations the current user is able to perform on this object + map can = 108816; + // Timestamp for the creation of this credential + string created_at = 464589301; + // EMail address + string email = 102965020; + // Has this credential been disabled? + bool is_disabled = 332107427; + // Timestamp for most recent login using credential + string logged_in_at = 411009336; + // Saml IdP's Unique ID for this user + string saml_user_id = 324077336; + // Short name for the type of this kind of credential + string type = 3120390; + // Link to get this item + string url = 107439; +} + +// WARNING: no writeable properties found for POST, PUT, or PATCH +message CredentialsTotp { + // Operations the current user is able to perform on this object + map can = 108816; + // Timestamp for the creation of this credential + string created_at = 464589301; + // Has this credential been disabled? + bool is_disabled = 332107427; + // Short name for the type of this kind of credential + string type = 3120390; + // User has verified + bool verified = 367216892; + // Link to get this item + string url = 107439; +} + + +message CustomWelcomeEmail { + // If true, custom email content will replace the default body of welcome emails + bool enabled = 387588912; + // The HTML to use as custom content for welcome emails. Script elements and other potentially dangerous markup will be removed. + string content = 294427041; + // The text to appear in the email subject line. Only available with a whitelabel license and whitelabel_configuration.advanced_custom_welcome_email enabled. + string subject = 373234517; + // The text to appear in the header line of the email body. Only available with a whitelabel license and whitelabel_configuration.advanced_custom_welcome_email enabled. + string header = 467446454; +} + + +message Dashboard { + // Operations the current user is able to perform on this object + map can = 108816; + // Content Favorite Id + string content_favorite_id = 363282886; + // Id of content metadata + string content_metadata_id = 4111259; + // Description + string description = 375083905; + // Is Hidden + bool hidden = 524703253; + // Unique Id + string id = 3205; + LookModel model = 102848809; + // Timezone in which the Dashboard will run by default. + string query_timezone = 391786756; + // Is Read-only + bool readonly = 373797887; + // Refresh Interval, as a time duration phrase like "2 hours 30 minutes". A number with no time units will be interpreted as whole seconds. + string refresh_interval = 382963810; + // Refresh Interval in milliseconds + int64 refresh_interval_to_i = 226412357; + FolderBase folder = 467441015; + // Dashboard Title + string title = 96607896; + // Id of User + string user_id = 413336787; + // Content Metadata Slug + string slug = 3184373; + // The preferred route for viewing this dashboard (ie: dashboards or dashboards-next) + string preferred_viewer = 351399076; + // Enables alerts to keep in sync with dashboard filter changes + bool alert_sync_with_dashboard_filter_enabled = 440126712; + // Background color + string background_color = 351188052; + // Time that the Dashboard was created. + google.protobuf.Timestamp created_at = 464589301; + // Enables crossfiltering in dashboards - only available in dashboards-next (beta) + bool crossfilter_enabled = 450959918; + // Elements + repeated DashboardElement dashboard_elements = 529192510; + // Filters + repeated DashboardFilter dashboard_filters = 470460328; + // Layouts + repeated DashboardLayout dashboard_layouts = 486259128; + // Whether or not a dashboard is 'soft' deleted. + bool deleted = 389460870; + // Time that the Dashboard was 'soft' deleted. + google.protobuf.Timestamp deleted_at = 464526556; + // Id of User that 'soft' deleted the dashboard. + string deleter_id = 180578797; + // Relative path of URI of LookML file to edit the dashboard (LookML dashboard only). + string edit_uri = 151398999; + // Number of times favorited + int64 favorite_count = 331565140; + // Sets the default state of the filters bar to collapsed or open + bool filters_bar_collapsed = 324566887; + // Time the dashboard was last accessed + google.protobuf.Timestamp last_accessed_at = 324585985; + // Time last viewed in the Looker web UI + google.protobuf.Timestamp last_viewed_at = 510619009; + // Time that the Dashboard was most recently updated. + google.protobuf.Timestamp updated_at = 464589803; + // Id of User that most recently updated the dashboard. + string last_updater_id = 397418835; + // Name of User that most recently updated the dashboard. + string last_updater_name = 176323850; + // Name of User that created the dashboard. + string user_name = 510631744; + // configuration option that governs how dashboard loading will happen. + string load_configuration = 446683328; + // Links this dashboard to a particular LookML dashboard such that calling a **sync** operation on that LookML dashboard will update this dashboard to match. + string lookml_link_id = 527451937; + // Show filters bar. **Security Note:** This property only affects the *cosmetic* appearance of the dashboard, not a user's ability to access data. Hiding the filters bar does **NOT** prevent users from changing filters by other means. For information on how to set up secure data access control policies, see [Control User Access to Data](https://looker.com/docs/r/api/control-access) + bool show_filters_bar = 70091731; + // Show title + bool show_title = 527232970; + // Id of folder + string folder_id = 271739380; + // Color of text on text tiles + string text_tile_text_color = 340996336; + // Tile background color + string tile_background_color = 467856637; + // Tile text color + string tile_text_color = 341304495; + // Title color + string title_color = 300393668; + // Number of times viewed in the Looker web UI + int64 view_count = 531861914; + DashboardAppearance appearance = 381592142; + // Relative URL of the dashboard + string url = 107439; +} + + +message DashboardAggregateTableLookml { + // Dashboard Id + string dashboard_id = 311679049; + // Aggregate Table LookML + string aggregate_table_lookml = 509161972; +} + + +message DashboardAppearance { + // Page margin (side) width + int64 page_side_margins = 400366988; + // Background color for the dashboard + string page_background_color = 467861694; + // Title alignment on dashboard tiles + string tile_title_alignment = 490534298; + // Space between tiles + int64 tile_space_between = 316877967; + // Background color for tiles + string tile_background_color = 467856637; + // Tile shadow on/off + bool tile_shadow = 333903371; + // Key color + string key_color = 312702017; +} + + +message DashboardBase { + // Operations the current user is able to perform on this object + map can = 108816; + // Content Favorite Id + string content_favorite_id = 363282886; + // Id of content metadata + string content_metadata_id = 4111259; + // Description + string description = 375083905; + // Is Hidden + bool hidden = 524703253; + // Unique Id + string id = 3205; + LookModel model = 102848809; + // Timezone in which the Dashboard will run by default. + string query_timezone = 391786756; + // Is Read-only + bool readonly = 373797887; + // Refresh Interval, as a time duration phrase like "2 hours 30 minutes". A number with no time units will be interpreted as whole seconds. + string refresh_interval = 382963810; + // Refresh Interval in milliseconds + int64 refresh_interval_to_i = 226412357; + FolderBase folder = 467441015; + // Dashboard Title + string title = 96607896; + // Id of User + string user_id = 413336787; + // Content Metadata Slug + string slug = 3184373; + // The preferred route for viewing this dashboard (ie: dashboards or dashboards-next) + string preferred_viewer = 351399076; +} + + +message DashboardElement { + // Operations the current user is able to perform on this object + map can = 108816; + // Text tile body text + string body_text = 491616469; + // Text tile body text as Html + string body_text_as_html = 271931937; + // Id of Dashboard + string dashboard_id = 311679049; + // Relative path of URI of LookML file to edit the dashboard element (LookML dashboard only). + string edit_uri = 151398999; + // Unique Id + string id = 3205; + LookWithQuery look = 3297857; + // Id Of Look + string look_id = 413287022; + // LookML link ID + string lookml_link_id = 527451937; + // ID of merge result + string merge_result_id = 3223242; + // Note Display + string note_display = 120563243; + // Note State + string note_state = 342745425; + // Note Text + string note_text = 491761577; + // Note Text as Html + string note_text_as_html = 271641721; + Query query = 115243016; + // Id Of Query + string query_id = 279007282; + // Refresh Interval + string refresh_interval = 382963810; + // Refresh Interval as integer + int64 refresh_interval_to_i = 226412357; + ResultMakerWithIdVisConfigAndDynamicFields result_maker = 476157982; + // ID of the ResultMakerLookup entry. + string result_maker_id = 388730214; + // Text tile subtitle text + string subtitle_text = 507090106; + // Title of dashboard element + string title = 96607896; + // Whether title is hidden + bool title_hidden = 308186743; + // Text tile title + string title_text = 424563514; + // Type + string type = 3120390; + // Count of Alerts associated to a dashboard element + int64 alert_count = 380995163; + // JSON with all the properties required for rich editor and buttons elements + string rich_content_json = 309924280; + // Text tile title text as Html + string title_text_as_html = 338316013; + // Text tile subtitle text as Html + string subtitle_text_as_html = 378999587; +} + + +message DashboardFilter { + // Operations the current user is able to perform on this object + map can = 108816; + // Unique Id + string id = 3205; + // Id of Dashboard + string dashboard_id = 311679049; + // Name of filter + string name = 3116757; + // Title of filter + string title = 96607896; + // Type of filter: one of date, number, string, or field + string type = 3120390; + // Default value of filter + string default_value = 429771091; + // Model of filter (required if type = field) + string model = 102848809; + // Explore of filter (required if type = field) + string explore = 370461451; + // Dimension of filter (required if type = field) + string dimension = 511293107; + // Field information + map field = 95669946; + // Display order of this filter relative to other filters + int64 row = 117914; + // Array of listeners for faceted filters + repeated string listens_to_filters = 493150066; + // Whether the filter allows multiple filter values (deprecated in the latest version of dashboards) + bool allow_multiple_values = 331288581; + // Whether the filter requires a value to run the dashboard + bool required = 497611616; + // The visual configuration for this filter. Used to set up how the UI for this filter should appear. + map ui_config = 413099149; +} + + +message DashboardLayout { + // Operations the current user is able to perform on this object + map can = 108816; + // Unique Id + string id = 3205; + // Id of Dashboard + string dashboard_id = 311679049; + // Type + string type = 3120390; + // Is Active + bool active = 322801217; + // Column Width + int64 column_width = 530224687; + // Width + int64 width = 99601414; + // Whether or not the dashboard layout is deleted. + bool deleted = 389460870; + // Title extracted from the dashboard this layout represents. + string dashboard_title = 310211885; + // Components + repeated DashboardLayoutComponent dashboard_layout_components = 438424664; +} + + +message DashboardLayoutComponent { + // Operations the current user is able to perform on this object + map can = 108816; + // Unique Id + string id = 3205; + // Id of Dashboard Layout + string dashboard_layout_id = 446158738; + // Id Of Dashboard Element + string dashboard_element_id = 426786746; + // Row + int64 row = 117914; + // Column + int64 column = 520752011; + // Width + int64 width = 99601414; + // Height + int64 height = 437383491; + // Whether or not the dashboard layout component is deleted + bool deleted = 389460870; + // Dashboard element title, extracted from the Dashboard Element. + string element_title = 502267269; + // Whether or not the dashboard element title is displayed. + bool element_title_hidden = 294103899; + // Visualization type, extracted from a query's vis_config + string vis_type = 174418393; +} + + +message DashboardLookml { + // Id of Dashboard + string dashboard_id = 311679049; + // (Write-Only) Id of the folder + string folder_id = 271739380; + // lookml of UDD + string lookml = 274764335; +} + + +message DataActionForm { + DataActionUserState state = 96828305; + // Array of form fields. + repeated DataActionFormField fields = 453472492; +} + + +message DataActionFormField { + // Name + string name = 3116757; + // Human-readable label + string label = 102846452; + // Description of field + string description = 375083905; + // Type of field. + string type = 3120390; + // Default value of the field. + string default = 305191039; + // The URL for an oauth link, if type is 'oauth_link'. + string oauth_url = 384133998; + // Whether or not a field supports interactive forms. + bool interactive = 382527960; + // Whether or not the field is required. This is a user-interface hint. A user interface displaying this form should not submit it without a value for this field. The action server must also perform this validation. + bool required = 497611616; + // If the form type is 'select', a list of options to be selected from. + repeated DataActionFormSelectOption options = 514074632; +} + + +message DataActionFormSelectOption { + // Name + string name = 3116757; + // Human-readable label + string label = 102846452; +} + + +message DataActionRequest { + // The JSON describing the data action. This JSON should be considered opaque and should be passed through unmodified from the query result it came from. + map action = 520003579; + // User input for any form values the data action might use. + map form_values = 318736160; +} + + +message DataActionResponse { + // ID of the webhook event that sent this data action. In some error conditions, this may be null. + string webhook_id = 363806496; + // Whether the data action was successful. + bool success = 480645727; + // When true, indicates that the client should refresh (rerun) the source query because the data may have been changed by the action. + bool refresh_query = 376320462; + ValidationError validation_errors = 491568280; + // Optional message returned by the data action server describing the state of the action that took place. This can be used to implement custom failure messages. If a failure is related to a particular form field, the server should send back a validation error instead. The Looker web UI does not currently display any message if the action indicates 'success', but may do so in the future. + string message = 452371230; +} + + +message DataActionUserState { + // User state data + string data = 3004310; + // Time in seconds until the state needs to be refreshed + int64 refresh_time = 387158452; +} + + +message Datagroup { + // Operations the current user is able to perform on this object + map can = 108816; + // UNIX timestamp at which this entry was created. + int64 created_at = 464589301; + // Unique ID of the datagroup + string id = 3205; + // Name of the model containing the datagroup. Unique when combined with name. + string model_name = 275154383; + // Name of the datagroup. Unique when combined with model_name. + string name = 3116757; + // UNIX timestamp before which cache entries are considered stale. Cannot be in the future. + int64 stale_before = 485547741; + // UNIX timestamp at which this entry trigger was last checked. + int64 trigger_check_at = 286685000; + // The message returned with the error of the last trigger check. + string trigger_error = 281624895; + // The value of the trigger when last checked. + string trigger_value = 293862133; + // UNIX timestamp at which this entry became triggered. Cannot be in the future. + int64 triggered_at = 311600238; +} + + +message DBConnection { + // Operations the current user is able to perform on this object + map can = 108816; + // Name of the connection. Also used as the unique identifier + string name = 3116757; + Dialect dialect = 373219958; + // SQL Runner snippets for this connection + repeated Snippet snippets = 357339282; + // True if PDTs are enabled on this connection + bool pdts_enabled = 376815834; + // Host name/address of server + string host = 3569816; + // Port number on server + string port = 3568863; + // Username for server authentication + string username = 382974997; + // (Write-Only) Password for server authentication + string password = 425146747; + // Whether the connection uses OAuth for authentication. + bool uses_oauth = 315217113; + // (Write-Only) Base64 encoded Certificate body for server authentication (when appropriate for dialect). + string certificate = 402617131; + // (Write-Only) Certificate keyfile type - .json or .p12 + string file_type = 278104344; + // Database name + string database = 446107481; + // Time zone of database + string db_timezone = 530059610; + // Timezone to use in queries + string query_timezone = 391786756; + // Scheme name + string schema = 353540960; + // Maximum number of concurrent connection to use + int64 max_connections = 481127633; + // Maximum size of query in GBs (BigQuery only, can be a user_attribute name) + string max_billing_gigabytes = 367356460; + // Use SSL/TLS when connecting to server + bool ssl = 107468; + // Verify the SSL + bool verify_ssl = 456329219; + // Name of temporary database (if used) + string tmp_db_name = 527957692; + // Additional params to add to JDBC connection string + string jdbc_additional_params = 453083961; + // Connection Pool Timeout, in seconds + int64 pool_timeout = 350051759; + // (Read/Write) SQL Dialect name + string dialect_name = 414841156; + // Creation date for this connection + string created_at = 464589301; + // Id of user who last modified this connection configuration + string user_id = 413336787; + // Is this an example connection? + bool example = 413170453; + // (Limited access feature) Are per user db credentials enabled. Enabling will remove previously set username and password + bool user_db_credentials = 401568052; + // Fields whose values map to user attribute names + repeated string user_attribute_fields = 424067660; + // Cron string specifying when maintenance such as PDT trigger checks and drops should be performed + string maintenance_cron = 291048785; + // Unix timestamp at start of last completed PDT trigger check process + string last_regen_at = 331462239; + // Unix timestamp at start of last completed PDT reap process + string last_reap_at = 467554102; + // Precache tables in the SQL Runner + bool sql_runner_precache_tables = 306935836; + // Fetch Information Schema For SQL Writing + bool sql_writing_with_info_schema = 462710732; + // SQL statements (semicolon separated) to issue after connecting to the database. Requires `custom_after_connect_statements` license feature + string after_connect_statements = 84658980; + DBConnectionOverride pdt_context_override = 442814086; + // Is this connection created and managed by Looker + bool managed = 386430087; + // The Id of the ssh tunnel this connection uses + string tunnel_id = 443214254; + // Maximum number of threads to use to build PDTs in parallel + int64 pdt_concurrency = 312616502; + // When disable_context_comment is true comment will not be added to SQL + bool disable_context_comment = 315519886; + // An External OAuth Application to use for authenticating to the database + string oauth_application_id = 254423058; + // When true, error PDTs will be retried every regenerator cycle + bool always_retry_failed_builds = 297083520; + // When true, query cost estimate will be displayed in explore. + bool cost_estimate_enabled = 468369479; + // PDT builds on this connection can be kicked off and cancelled via API. + bool pdt_api_control_enabled = 521873634; +} + + +message DBConnectionBase { + // Operations the current user is able to perform on this object + map can = 108816; + // Name of the connection. Also used as the unique identifier + string name = 3116757; + Dialect dialect = 373219958; + // SQL Runner snippets for this connection + repeated Snippet snippets = 357339282; + // True if PDTs are enabled on this connection + bool pdts_enabled = 376815834; +} + + +message DBConnectionOverride { + // Context in which to override (`pdt` is the only allowed value) + string context = 445708328; + // Host name/address of server + string host = 3569816; + // Port number on server + string port = 3568863; + // Username for server authentication + string username = 382974997; + // (Write-Only) Password for server authentication + string password = 425146747; + // Whether or not the password is overridden in this context + bool has_password = 269826504; + // (Write-Only) Base64 encoded Certificate body for server authentication (when appropriate for dialect). + string certificate = 402617131; + // (Write-Only) Certificate keyfile type - .json or .p12 + string file_type = 278104344; + // Database name + string database = 446107481; + // Scheme name + string schema = 353540960; + // Additional params to add to JDBC connection string + string jdbc_additional_params = 453083961; + // SQL statements (semicolon separated) to issue after connecting to the database. Requires `custom_after_connect_statements` license feature + string after_connect_statements = 84658980; +} + + +message DBConnectionTestResult { + // Operations the current user is able to perform on this object + map can = 108816; + // JDBC connection string. (only populated in the 'connect' test) + string connection_string = 493767698; + // Result message of test + string message = 452371230; + // Name of test + string name = 3116757; + // Result code of test + string status = 445505145; +} + + +message DelegateOauthTest { + // Delegate Oauth Connection Name + string name = 3116757; + // The ID of the installation target. For Slack, this would be workspace id. + string installation_target_id = 481962692; + // Installation ID + string installation_id = 365386880; + // Whether or not the test was successful + bool success = 480645727; +} + + +message DependencyGraph { + // The graph structure in the dot language that can be rendered into an image. + string graph_text = 423127647; +} + +// Status of the dependencies in your project. Valid values are: "lock_optional", "lock_required", "lock_error", "install_none". (Enum defined in ProjectWorkspace) +enum DependencyStatus { + _DEPENDENCY_STATUS_UNSET = 0; + DEPENDENCY_STATUS_LOCK_OPTIONAL = 149493915; + DEPENDENCY_STATUS_LOCK_REQUIRED = 356009478; + DEPENDENCY_STATUS_LOCK_ERROR = 438670100; + DEPENDENCY_STATUS_INSTALL_NONE = 509733909; +} + +// Type of destination that the alert will be sent to Valid values are: "EMAIL", "ACTION_HUB". (Enum defined in AlertDestination) +enum DestinationType { + _DESTINATION_TYPE_UNSET = 0; + DESTINATION_TYPE_EMAIL = 300333306; + DESTINATION_TYPE_ACTION_HUB = 325002464; +} + + +message Dialect { + // The name of the dialect + string name = 3116757; + // The human-readable label of the connection + string label = 102846452; + // Whether the dialect supports query cost estimates + bool supports_cost_estimate = 459772799; + // How the dialect handles cost estimation + string cost_estimate_style = 227310836; + // PDT index columns + string persistent_table_indexes = 491969169; + // PDT sortkey columns + string persistent_table_sortkeys = 489464053; + // PDT distkey column + string persistent_table_distkey = 305664928; + // Suports streaming results + bool supports_streaming = 302473166; + // Should SQL Runner snippets automatically be run + bool automatically_run_sql_runner_snippets = 381066281; + // Array of names of the tests that can be run on a connection using this dialect + repeated string connection_tests = 291539296; + // Is supported with the inducer (i.e. generate from sql) + bool supports_inducer = 271124543; + // Can multiple databases be accessed from a connection using this dialect + bool supports_multiple_databases = 518173666; + // Whether the dialect supports allowing Looker to build persistent derived tables + bool supports_persistent_derived_tables = 391774333; + // Does the database have client SSL support settable through the JDBC string explicitly? + bool has_ssl_support = 281159483; +} + + +message DialectInfo { + // Operations the current user is able to perform on this object + map can = 108816; + // Default number max connections + string default_max_connections = 238864516; + // Default port number + string default_port = 336218799; + // Is the supporting driver installed + bool installed = 175385798; + // The human-readable label of the connection + string label = 102846452; + // What the dialect calls the equivalent of a normal SQL table + string label_for_database_equivalent = 299647913; + // The name of the dialect + string name = 3116757; + DialectInfoOptions supported_options = 307677965; +} + + +message DialectInfoOptions { + // Has additional params support + bool additional_params = 468924663; + // Has auth support + bool auth = 3213464; + // Has host support + bool host = 3569816; + // Has support for a service account + bool oauth_credentials = 532985883; + // Has project name support + bool project_name = 414886993; + // Has schema support + bool schema = 353540960; + // Has SSL support + bool ssl = 107468; + // Has timezone support + bool timezone = 516241033; + // Has tmp table support + bool tmp_table = 521896499; + // Username is required + bool username_required = 309785338; +} + + +message DigestEmails { + // Whether or not digest emails are enabled + bool is_enabled = 358266715; +} + + +message DigestEmailSend { + // True if content was successfully generated and delivered + bool configuration_delivered = 349204366; +} + + +message DiscretePalette { + // Unique identity string + string id = 3205; + // Label for palette + string label = 102846452; + // Type of palette + string type = 3120390; + // Array of colors in the palette + repeated string colors = 446959704; +} + + +message EgressIpAddresses { + // Egress IP addresses + repeated string egress_ip_addresses = 342957515; +} + + +message EmbedParams { + // The complete URL of the Looker UI page to display in the embed context. For example, to display the dashboard with id 34, `target_url` would look like: `https://mycompany.looker.com:9999/dashboards/34`. `target_uri` MUST contain a scheme (HTTPS), domain name, and URL path. Port must be included if it is required to reach the Looker server from browser clients. If the Looker instance is behind a load balancer or other proxy, `target_uri` must be the public-facing domain name and port required to reach the Looker instance, not the actual internal network machine name of the Looker instance. + string target_url = 58679585; + // Number of seconds the SSO embed session will be valid after the embed session is started. Defaults to 300 seconds. Maximum session length accepted is 2592000 seconds (30 days). + int64 session_length = 520857231; + // When true, the embed session will purge any residual Looker login state (such as in browser cookies) before creating a new login state with the given embed user info. Defaults to true. + bool force_logout_login = 405871388; +} + + +message EmbedSecret { + // Signing algorithm to use with this secret. Either `hmac/sha-256`(default) or `hmac/sha-1` + string algorithm = 110603631; + // When secret was created + string created_at = 464589301; + // Is this secret currently enabled + bool enabled = 387588912; + // Unique Id + string id = 3205; + // Secret for use with SSO embedding + string secret = 438607800; + // Id of user who created this secret + string user_id = 413336787; +} + + +message EmbedSsoParams { + // The complete URL of the Looker UI page to display in the embed context. For example, to display the dashboard with id 34, `target_url` would look like: `https://mycompany.looker.com:9999/dashboards/34`. `target_uri` MUST contain a scheme (HTTPS), domain name, and URL path. Port must be included if it is required to reach the Looker server from browser clients. If the Looker instance is behind a load balancer or other proxy, `target_uri` must be the public-facing domain name and port required to reach the Looker instance, not the actual internal network machine name of the Looker instance. + string target_url = 58679585; + // Number of seconds the SSO embed session will be valid after the embed session is started. Defaults to 300 seconds. Maximum session length accepted is 2592000 seconds (30 days). + int64 session_length = 520857231; + // When true, the embed session will purge any residual Looker login state (such as in browser cookies) before creating a new login state with the given embed user info. Defaults to true. + bool force_logout_login = 405871388; + // A value from an external system that uniquely identifies the embed user. Since the user_ids of Looker embed users may change with every embed session, external_user_id provides a way to assign a known, stable user identifier across multiple embed sessions. + string external_user_id = 459767842; + // First name of the embed user. Defaults to 'Embed' if not specified + string first_name = 277109009; + // Last name of the embed user. Defaults to 'User' if not specified + string last_name = 510613627; + // Sets the user timezone for the embed user session, if the User Specific Timezones setting is enabled in the Looker admin settings. A value of `null` forces the embed user to use the Looker Application Default Timezone. You MUST omit this property from the request if the User Specific Timezones setting is disabled. Timezone values are validated against the IANA Timezone standard and can be seen in the Application Time Zone dropdown list on the Looker General Settings admin page. + string user_timezone = 428335428; + // List of Looker permission names to grant to the embed user. Requested permissions will be filtered to permissions allowed for embed sessions. + repeated string permissions = 322818529; + // List of model names that the embed user may access + repeated string models = 449883061; + // List of Looker group ids in which to enroll the embed user + repeated string group_ids = 496513282; + // A unique value identifying an embed-exclusive group. Multiple embed users using the same `external_group_id` value will be able to share Looker content with each other. Content and embed users associated with the `external_group_id` will not be accessible to normal Looker users or embed users not associated with this `external_group_id`. + string external_group_id = 488769304; + // A dictionary of name-value pairs associating a Looker user attribute name with a value. + map user_attributes = 15243445; + // Id of the embed secret to use to sign this SSO url. If specified, the value must be an id of a valid (active) secret defined in the Looker instance. If not specified, the URL will be signed with the newest active embed secret defined in the Looker instance. + string secret_id = 214072950; +} + + +message EmbedUrlResponse { + // The embed URL. Any modification to this string will make the URL unusable. + string url = 107439; +} + + +message Error { + // Error details + string message = 452371230; + // Documentation link + string documentation_url = 402392181; +} + + +message ExternalOauthApplication { + // Operations the current user is able to perform on this object + map can = 108816; + // ID of this OAuth Application + string id = 3205; + // The name of this application. For Snowflake connections, this should be the name of the host database. + string name = 3116757; + // The OAuth Client ID for this application + string client_id = 206142577; + // (Write-Only) The OAuth Client Secret for this application + string client_secret = 509513545; + // The database dialect for this application. + string dialect_name = 414841156; + // Creation time for this application + google.protobuf.Timestamp created_at = 464589301; +} + +// The style of dimension fill that is possible for this field. Null if no dimension fill is possible. Valid values are: "enumeration", "range". (Enum defined in LookmlModelExploreField) +enum FillStyle { + _FILL_STYLE_UNSET = 0; + FILL_STYLE_ENUMERATION = 502983325; + FILL_STYLE_RANGE = 298945796; +} + + +message Folder { + // Unique Name + string name = 3116757; + // Id of Parent. If the parent id is null, this is a root-level entry + string parent_id = 206134256; + // Unique Id + string id = 3205; + // Id of content metadata + string content_metadata_id = 4111259; + // Time the space was created + google.protobuf.Timestamp created_at = 464589301; + // User Id of Creator + string creator_id = 466744818; + // Children Count + int64 child_count = 377257323; + // Embedder's Id if this folder was autogenerated as an embedding shared folder via 'external_group_id' in an SSO embed login + string external_id = 21038031; + // Folder is an embed folder + bool is_embed = 270178686; + // Folder is the root embed shared folder + bool is_embed_shared_root = 316207960; + // Folder is the root embed users folder + bool is_embed_users_root = 446858283; + // Folder is a user's personal folder + bool is_personal = 352024274; + // Folder is descendant of a user's personal folder + bool is_personal_descendant = 356953100; + // Folder is the root shared folder + bool is_shared_root = 394142236; + // Folder is the root user folder + bool is_users_root = 326190292; + // Operations the current user is able to perform on this object + map can = 108816; + // Dashboards + repeated DashboardBase dashboards = 301014335; + // Looks + repeated LookWithDashboards looks = 109502772; +} + + +message FolderBase { + // Unique Name + string name = 3116757; + // Id of Parent. If the parent id is null, this is a root-level entry + string parent_id = 206134256; + // Unique Id + string id = 3205; + // Id of content metadata + string content_metadata_id = 4111259; + // Time the folder was created + google.protobuf.Timestamp created_at = 464589301; + // User Id of Creator + string creator_id = 466744818; + // Children Count + int64 child_count = 377257323; + // Embedder's Id if this folder was autogenerated as an embedding shared folder via 'external_group_id' in an SSO embed login + string external_id = 21038031; + // Folder is an embed folder + bool is_embed = 270178686; + // Folder is the root embed shared folder + bool is_embed_shared_root = 316207960; + // Folder is the root embed users folder + bool is_embed_users_root = 446858283; + // Folder is a user's personal folder + bool is_personal = 352024274; + // Folder is descendant of a user's personal folder + bool is_personal_descendant = 356953100; + // Folder is the root shared folder + bool is_shared_root = 394142236; + // Folder is the root user folder + bool is_users_root = 326190292; + // Operations the current user is able to perform on this object + map can = 108816; +} + +// Specifies the data format of the region information. Valid values are: "topojson", "vector_tile_region". (Enum defined in LookmlModelExploreFieldMapLayer) +enum Format { + _FORMAT_UNSET = 0; + FORMAT_TOPOJSON = 148421078; + FORMAT_VECTOR_TILE_REGION = 427793171; +} + + +message GitBranch { + // Operations the current user is able to perform on this object + map can = 108816; + // The short name on the local. Updating `name` results in `git checkout ` + string name = 3116757; + // The name of the remote + string remote = 323219953; + // The short name on the remote + string remote_name = 426820380; + // Name of error + string error = 108701384; + // Message describing an error if present + string message = 452371230; + // Name of the owner of a personal branch + string owner_name = 276542130; + // Whether or not this branch is readonly + bool readonly = 373797887; + // Whether or not this branch is a personal branch - readonly for all developers except the owner + bool personal = 401561200; + // Whether or not a local ref exists for the branch + bool is_local = 480347957; + // Whether or not a remote ref exists for the branch + bool is_remote = 271066654; + // Whether or not this is the production branch + bool is_production = 372362185; + // Number of commits the local branch is ahead of the remote + int64 ahead_count = 377174436; + // Number of commits the local branch is behind the remote + int64 behind_count = 462885002; + // UNIX timestamp at which this branch was last committed. + int64 commit_at = 122599045; + // The resolved ref of this branch. Updating `ref` results in `git reset --hard ``. + string ref = 101267; + // The resolved ref of this branch remote. + string remote_ref = 448414617; +} + + +message GitConnectionTest { + // Operations the current user is able to perform on this object + map can = 108816; + // Human readable string describing the test + string description = 375083905; + // A short string, uniquely naming this test + string id = 3205; +} + + +message GitConnectionTestResult { + // Operations the current user is able to perform on this object + map can = 108816; + // A short string, uniquely naming this test + string id = 3205; + // Additional data from the test + string message = 452371230; + // Either 'pass' or 'fail' + string status = 445505145; +} + + +message GitStatus { + // Git action: add, delete, etc + string action = 520003579; + // When true, changes to the local file conflict with the remote repository + bool conflict = 270009051; + // When true, the file can be reverted to an earlier state + bool revertable = 496231983; + // Git description of the action + string text = 3574323; +} + + +message Group { + // Operations the current user is able to perform on this object + map can = 108816; + // Group can be used in content access controls + bool can_add_to_content_metadata = 454846998; + // Currently logged in user is group member + bool contains_current_user = 413863500; + // External Id group if embed group + string external_group_id = 488769304; + // Group membership controlled outside of Looker + bool externally_managed = 38023048; + // Unique Id + string id = 3205; + // New users are added to this group by default + bool include_by_default = 310734864; + // Name of group + string name = 3116757; + // Number of users included in this group + int64 user_count = 531824753; +} + + +message GroupHierarchy { + // Operations the current user is able to perform on this object + map can = 108816; + // Group can be used in content access controls + bool can_add_to_content_metadata = 454846998; + // Currently logged in user is group member + bool contains_current_user = 413863500; + // External Id group if embed group + string external_group_id = 488769304; + // Group membership controlled outside of Looker + bool externally_managed = 38023048; + // Unique Id + string id = 3205; + // New users are added to this group by default + bool include_by_default = 310734864; + // Name of group + string name = 3116757; + // Number of users included in this group + int64 user_count = 531824753; + // IDs of parents of this group + repeated string parent_group_ids = 357424798; + // Role IDs assigned to group + repeated string role_ids = 212692081; +} + +// WARNING: no writeable properties found for POST, PUT, or PATCH +message GroupIdForGroupInclusion { + // Id of group + string group_id = 287220091; +} + +// WARNING: no writeable properties found for POST, PUT, or PATCH +message GroupIdForGroupUserInclusion { + // Id of user + string user_id = 413336787; +} + + +message GroupSearch { + // Operations the current user is able to perform on this object + map can = 108816; + // Group can be used in content access controls + bool can_add_to_content_metadata = 454846998; + // Currently logged in user is group member + bool contains_current_user = 413863500; + // External Id group if embed group + string external_group_id = 488769304; + // Group membership controlled outside of Looker + bool externally_managed = 38023048; + // Unique Id + string id = 3205; + // New users are added to this group by default + bool include_by_default = 310734864; + // Name of group + string name = 3116757; + // Number of users included in this group + int64 user_count = 531824753; + // Roles assigned to group + repeated Role roles = 109321149; +} + + +message HomepageItem { + // Operations the current user is able to perform on this object + map can = 108816; + // Name of user who created the content this item is based on + string content_created_by = 345901194; + // Content favorite id associated with the item this content is based on + string content_favorite_id = 363282886; + // Content metadata id associated with the item this content is based on + string content_metadata_id = 4111259; + // Last time the content that this item is based on was updated + string content_updated_at = 408537127; + // Custom description entered by the user, if present + string custom_description = 517484699; + // (Write-Only) base64 encoded image data + string custom_image_data_base64 = 427605623; + // Custom image_url entered by the user, if present + string custom_image_url = 497368103; + // Custom title entered by the user, if present + string custom_title = 413477701; + // Custom url entered by the user, if present + string custom_url = 249459585; + // Dashboard to base this item on + string dashboard_id = 311679049; + // The actual description for display + string description = 375083905; + // Number of times content has been favorited, if present + int64 favorite_count = 331565140; + // Associated Homepage Section + string homepage_section_id = 365080154; + // Unique Id + string id = 3205; + // The actual image_url for display + string image_url = 384928173; + // The container folder name of the content + string location = 320111754; + // Look to base this item on + string look_id = 413287022; + // LookML Dashboard to base this item on + string lookml_dashboard_id = 351003409; + // An arbitrary integer representing the sort order within the section + int64 order = 108390030; + // Number of seconds it took to fetch the section this item is in + float section_fetch_time = 460289958; + // The actual title for display + string title = 96607896; + // The actual url for display + string url = 107439; + // Whether the custom description should be used instead of the content description, if the item is associated with content + bool use_custom_description = 532957353; + // Whether the custom image should be used instead of the content image, if the item is associated with content + bool use_custom_image = 398202105; + // Whether the custom title should be used instead of the content title, if the item is associated with content + bool use_custom_title = 533570657; + // Whether the custom url should be used instead of the content url, if the item is associated with content + bool use_custom_url = 441644771; + // Number of times content has been viewed, if present + int64 view_count = 531861914; +} + + +message HomepageSection { + // Operations the current user is able to perform on this object + map can = 108816; + // Time at which this section was created. + google.protobuf.Timestamp created_at = 464589301; + // Time at which this section was deleted. + google.protobuf.Timestamp deleted_at = 464526556; + // A URL pointing to a page showing further information about the content in the section. + string detail_url = 284195425; + // Id reference to parent homepage + string homepage_id = 340572339; + // Items in the homepage section + repeated HomepageItem homepage_items = 348133075; + // Unique Id + string id = 3205; + // Is this a header section (has no items) + bool is_header = 367089376; + // ids of the homepage items in the order they should be displayed + repeated string item_order = 377743503; + // Name of row + string title = 96607896; + // Time at which this section was last updated. + google.protobuf.Timestamp updated_at = 464589803; + // Description of the content found in this section. + string description = 375083905; + // ids of the homepage items the user can see in the order they should be displayed + repeated string visible_item_order = 411259595; +} + + +message ImportedProject { + // Dependency name + string name = 3116757; + // Url for a remote dependency + string url = 107439; + // Ref for a remote dependency + string ref = 101267; + // Flag signifying if a dependency is remote or local + bool is_remote = 271066654; +} + + +message Integration { + // Operations the current user is able to perform on this object + map can = 108816; + // ID of the integration. + string id = 3205; + // ID of the integration hub. + string integration_hub_id = 469183532; + // Label for the integration. + string label = 102846452; + // Description of the integration. + string description = 375083905; + // Whether the integration is available to users. + bool enabled = 387588912; + // Array of params for the integration. + repeated IntegrationParam params = 449474371; + // A list of data formats the integration supports. If unspecified, the default is all data formats. Valid values are: "txt", "csv", "inline_json", "json", "json_label", "json_detail", "json_detail_lite_stream", "xlsx", "html", "wysiwyg_pdf", "assembled_pdf", "wysiwyg_png", "csv_zip". + repeated SupportedFormats supported_formats = 331633402; + // A list of action types the integration supports. Valid values are: "cell", "query", "dashboard", "none". + repeated SupportedActionTypes supported_action_types = 274307823; + // A list of formatting options the integration supports. If unspecified, defaults to all formats. Valid values are: "formatted", "unformatted". + repeated SupportedFormattings supported_formattings = 50364637; + // A list of visualization formatting options the integration supports. If unspecified, defaults to all formats. Valid values are: "apply", "noapply". + repeated SupportedVisualizationFormattings supported_visualization_formattings = 401393278; + // A list of all the download mechanisms the integration supports. The order of values is not significant: Looker will select the most appropriate supported download mechanism for a given query. The integration must ensure it can handle any of the mechanisms it claims to support. If unspecified, this defaults to all download setting values. Valid values are: "push", "url". + repeated SupportedDownloadSettings supported_download_settings = 391412410; + // URL to an icon for the integration. + string icon_url = 390945851; + // Whether the integration uses oauth. + bool uses_oauth = 315217113; + // A list of descriptions of required fields that this integration is compatible with. If there are multiple entries in this list, the integration requires more than one field. If unspecified, no fields will be required. + repeated IntegrationRequiredField required_fields = 396888615; + // Whether the integration uses delegate oauth, which allows federation between an integration installation scope specific entity (like org, group, and team, etc.) and Looker. + bool delegate_oauth = 356926112; + // Whether the integration is available to users. + repeated string installed_delegate_oauth_targets = 359534361; +} + + +message IntegrationHub { + // Operations the current user is able to perform on this object + map can = 108816; + // ID of the hub. + string id = 3205; + // URL of the hub. + string url = 107439; + // Label of the hub. + string label = 102846452; + // Whether this hub is a first-party integration hub operated by Looker. + bool official = 440195554; + // An error message, present if the integration hub metadata could not be fetched. If this is present, the integration hub is unusable. + string fetch_error_message = 199155787; + // (Write-Only) An authorization key that will be sent to the integration hub on every request. + string authorization_token = 290870323; + // Whether the authorization_token is set for the hub. + bool has_authorization_token = 416501196; + // Whether the legal agreement message has been signed by the user. This only matters if legal_agreement_required is true. + bool legal_agreement_signed = 325052468; + // Whether the legal terms for the integration hub are required before use. + bool legal_agreement_required = 426842657; + // The legal agreement text for this integration hub. + string legal_agreement_text = 156768264; +} + + +message IntegrationParam { + // Name of the parameter. + string name = 3116757; + // Label of the parameter. + string label = 102846452; + // Short description of the parameter. + string description = 375083905; + // Whether the parameter is required to be set to use the destination. If unspecified, this defaults to false. + bool required = 497611616; + // Whether the parameter has a value set. + bool has_value = 163831412; + // The current value of the parameter. Always null if the value is sensitive. When writing, null values will be ignored. Set the value to an empty string to clear it. + string value = 96868081; + // When present, the param's value comes from this user attribute instead of the 'value' parameter. Set to null to use the 'value'. + string user_attribute_name = 324327856; + // Whether the parameter contains sensitive data like API credentials. If unspecified, this defaults to true. + bool sensitive = 275496437; + // When true, this parameter must be assigned to a user attribute in the admin panel (instead of a constant value), and that value may be updated by the user as part of the integration flow. + bool per_user = 336178893; + // When present, the param represents the oauth url the user will be taken to. + string delegate_oauth_url = 409466811; +} + + +message IntegrationRequiredField { + // Matches a field that has this tag. + string tag = 102106; + // If present, supercedes 'tag' and matches a field that has any of the provided tags. + repeated string any_tag = 189295353; + // If present, supercedes 'tag' and matches a field that has all of the provided tags. + repeated string all_tags = 407395573; +} + + +message IntegrationTestResult { + // Whether or not the test was successful + bool success = 480645727; + // A message representing the results of the test. + string message = 452371230; + // An array of connection test result for delegate oauth actions. + repeated DelegateOauthTest delegate_oauth_result = 346098944; +} + + +message InternalHelpResources { + // Operations the current user is able to perform on this object + map can = 108816; + // If true and internal help resources content is not blank then the link for internal help resources will be shown in the help menu and the content displayed within Looker + bool enabled = 387588912; +} + + +message InternalHelpResourcesContent { + // Operations the current user is able to perform on this object + map can = 108816; + // Text to display in the help menu item which will display the internal help resources + string organization_name = 71465175; + // Content to be displayed in the internal help resources page/modal + string markdown_content = 379297421; +} + +// The type of the investigative content Valid values are: "dashboard". (Enum defined in Alert) +enum InvestigativeContentType { + _INVESTIGATIVE_CONTENT_TYPE_UNSET = 0; + INVESTIGATIVE_CONTENT_TYPE_DASHBOARD = 468222720; +} + + +message LDAPConfig { + // Operations the current user is able to perform on this object + map can = 108816; + // Allow alternate email-based login via '/login/email' for admins and for specified users with the 'login_special_email' permission. This option is useful as a fallback during ldap setup, if ldap config problems occur later, or if you need to support some users who are not in your ldap directory. Looker email/password logins are always disabled for regular users when ldap is enabled. + bool alternate_email_login_allowed = 415381803; + // (Write-Only) Password for the LDAP account used to access the LDAP server + string auth_password = 450780407; + // Users will not be allowed to login at all unless a role for them is found in LDAP if set to true + bool auth_requires_role = 359416088; + // Distinguished name of LDAP account used to access the LDAP server + string auth_username = 358003939; + // LDAP server hostname + string connection_host = 526430378; + // LDAP host port + string connection_port = 325512703; + // Use Transport Layer Security + bool connection_tls = 364401289; + // Do not verify peer when using TLS + bool connection_tls_no_verify = 374286647; + // (Write-Only) Array of ids of groups that will be applied to new users the first time they login via LDAP + repeated string default_new_user_group_ids = 417958896; + // (Read-only) Groups that will be applied to new users the first time they login via LDAP + repeated Group default_new_user_groups = 487231946; + // (Write-Only) Array of ids of roles that will be applied to new users the first time they login via LDAP + repeated string default_new_user_role_ids = 319694385; + // (Read-only) Roles that will be applied to new users the first time they login via LDAP + repeated Role default_new_user_roles = 478070081; + // Enable/Disable LDAP authentication for the server + bool enabled = 387588912; + // Don't attempt to do LDAP search result paging (RFC 2696) even if the LDAP server claims to support it. + bool force_no_page = 272379335; + // (Read-only) Array of mappings between LDAP Groups and Looker Roles + repeated LDAPGroupRead groups = 447792362; + // Base dn for finding groups in LDAP searches + string groups_base_dn = 321940973; + // Identifier for a strategy for how Looker will search for groups in the LDAP server + string groups_finder_type = 21587272; + // LDAP Group attribute that signifies the members of the groups. Most commonly 'member' + string groups_member_attribute = 450619343; + // Optional comma-separated list of supported LDAP objectclass for groups when doing groups searches + string groups_objectclasses = 402805559; + // LDAP Group attribute that signifies the user in a group. Most commonly 'dn' + string groups_user_attribute = 394608153; + // (Read/Write) Array of mappings between LDAP Groups and arrays of Looker Role ids + repeated LDAPGroupWrite groups_with_role_ids = 475375615; + // (Read-only) Has the password been set for the LDAP account used to access the LDAP server + bool has_auth_password = 433230825; + // Merge first-time ldap login to existing user account by email addresses. When a user logs in for the first time via ldap this option will connect this user into their existing account by finding the account with a matching email address. Otherwise a new user account will be created for the user. + bool merge_new_users_by_email = 469551208; + // When this config was last modified + string modified_at = 475018772; + // User id of user who last modified this config + string modified_by = 358701837; + // Set user roles in Looker based on groups from LDAP + bool set_roles_from_groups = 390938262; + // (Write-Only) Test LDAP user password. For ldap tests only. + string test_ldap_password = 494187581; + // (Write-Only) Test LDAP user login id. For ldap tests only. + string test_ldap_user = 507373170; + // Name of user record attributes used to indicate email address field + string user_attribute_map_email = 398964231; + // Name of user record attributes used to indicate first name + string user_attribute_map_first_name = 400740186; + // Name of user record attributes used to indicate last name + string user_attribute_map_last_name = 521517530; + // Name of user record attributes used to indicate unique record id + string user_attribute_map_ldap_id = 466191369; + // (Read-only) Array of mappings between LDAP User Attributes and Looker User Attributes + repeated LDAPUserAttributeRead user_attributes = 15243445; + // (Read/Write) Array of mappings between LDAP User Attributes and arrays of Looker User Attribute ids + repeated LDAPUserAttributeWrite user_attributes_with_ids = 332652790; + // Distinguished name of LDAP node used as the base for user searches + string user_bind_base_dn = 289800939; + // (Optional) Custom RFC-2254 filter clause for use in finding user during login. Combined via 'and' with the other generated filter clauses. + string user_custom_filter = 129489070; + // Name(s) of user record attributes used for matching user login id (comma separated list) + string user_id_attribute_names = 236102283; + // (Optional) Name of user record objectclass used for finding user during login id + string user_objectclass = 358011678; + // Allow LDAP auth'd users to be members of non-reflected Looker groups. If 'false', user will be removed from non-reflected groups on login. + bool allow_normal_group_membership = 215101880; + // LDAP auth'd users will be able to inherit roles from non-reflected Looker groups. + bool allow_roles_from_normal_groups = 301477131; + // Allows roles to be directly assigned to LDAP auth'd users. + bool allow_direct_roles = 276166815; + // Link to get this item + string url = 107439; +} + + +message LDAPConfigTestIssue { + // Severity of the issue. Error or Warning + string severity = 441317294; + // Message describing the issue + string message = 452371230; +} + + +message LDAPConfigTestResult { + // Additional details for error cases + string details = 529834159; + // Array of issues/considerations about the result + repeated LDAPConfigTestIssue issues = 452869789; + // Short human readable test about the result + string message = 452371230; + // Test status code: always 'success' or 'error' + string status = 445505145; + // A more detailed trace of incremental results during auth tests + string trace = 96321797; + LDAPUser user = 3496917; + // Link to ldap config + string url = 107439; +} + + +message LDAPGroupRead { + // Unique Id + string id = 3205; + // Unique Id of group in Looker + string looker_group_id = 384327945; + // Name of group in Looker + string looker_group_name = 202505630; + // Name of group in LDAP + string name = 3116757; + // Looker Roles + repeated Role roles = 109321149; + // Link to ldap config + string url = 107439; +} + + +message LDAPGroupWrite { + // Unique Id + string id = 3205; + // Unique Id of group in Looker + string looker_group_id = 384327945; + // Name of group in Looker + string looker_group_name = 202505630; + // Name of group in LDAP + string name = 3116757; + // Looker Role Ids + repeated string role_ids = 212692081; + // Link to ldap config + string url = 107439; +} + + +message LDAPUser { + // Array of user's email addresses and aliases for use in migration + repeated string all_emails = 334744309; + // Dictionary of user's attributes (name/value) + map attributes = 339449220; + // Primary email address + string email = 102965020; + // First name + string first_name = 277109009; + // Array of user's groups (group names only) + repeated string groups = 447792362; + // Last Name + string last_name = 510613627; + // LDAP's distinguished name for the user record + string ldap_dn = 448809928; + // LDAP's Unique ID for the user + string ldap_id = 413320812; + // Array of user's roles (role names only) + repeated string roles = 109321149; + // Link to ldap config + string url = 107439; +} + + +message LDAPUserAttributeRead { + // Name of User Attribute in LDAP + string name = 3116757; + // Required to be in LDAP assertion for login to be allowed to succeed + bool required = 497611616; + // Looker User Attributes + repeated UserAttribute user_attributes = 15243445; + // Link to ldap config + string url = 107439; +} + + +message LDAPUserAttributeWrite { + // Name of User Attribute in LDAP + string name = 3116757; + // Required to be in LDAP assertion for login to be allowed to succeed + bool required = 497611616; + // Looker User Attribute Ids + repeated string user_attribute_ids = 329947583; + // Link to ldap config + string url = 107439; +} + + +message LegacyFeature { + // Operations the current user is able to perform on this object + map can = 108816; + // Unique Id + string id = 3205; + // Name + string name = 3116757; + // Description + string description = 375083905; + // Whether this feature has been enabled by a user + bool enabled_locally = 276773134; + // Whether this feature is currently enabled + bool enabled = 387588912; + // Looker version where this feature became a legacy feature + string disallowed_as_of_version = 108908664; + // Looker version where this feature will be automatically disabled + string disable_on_upgrade_to_version = 413856792; + // Future Looker version where this feature will be removed + string end_of_life_version = 58310631; + // URL for documentation about this feature + string documentation_url = 402392181; + // Approximate date that this feature will be automatically disabled. + google.protobuf.Timestamp approximate_disable_date = 314149644; + // Approximate date that this feature will be removed. + google.protobuf.Timestamp approximate_end_of_life_date = 421324170; + // Whether this legacy feature may have been automatically disabled when upgrading to the current version. + bool has_disabled_on_upgrade = 505654449; +} + + +message Locale { + // Code for Locale + string code = 3108531; + // Name of Locale in its own language + string native_name = 425145485; + // Name of Locale in English + string english_name = 503296078; +} + + +message LocalizationSettings { + // Default locale for localization + string default_locale = 150259352; + // Localization level - strict or permissive + string localization_level = 522116983; +} + + +message Look { + // Operations the current user is able to perform on this object + map can = 108816; + // Id of content metadata + string content_metadata_id = 4111259; + // Unique Id + string id = 3205; + // Look Title + string title = 96607896; + // User Id + string user_id = 413336787; + // Content Favorite Id + string content_favorite_id = 363282886; + // Time that the Look was created. + google.protobuf.Timestamp created_at = 464589301; + // Whether or not a look is 'soft' deleted. + bool deleted = 389460870; + // Time that the Look was deleted. + google.protobuf.Timestamp deleted_at = 464526556; + // Id of User that deleted the look. + string deleter_id = 180578797; + // Description + string description = 375083905; + // Embed Url + string embed_url = 385173709; + // Excel File Url + string excel_file_url = 494304189; + // Number of times favorited + int64 favorite_count = 331565140; + // Google Spreadsheet Formula + string google_spreadsheet_formula = 494865843; + // Image Embed Url + string image_embed_url = 437097913; + // auto-run query when Look viewed + bool is_run_on_load = 487054535; + // Time that the Look was last accessed by any user + google.protobuf.Timestamp last_accessed_at = 324585985; + // Id of User that last updated the look. + string last_updater_id = 397418835; + // Time last viewed in the Looker web UI + google.protobuf.Timestamp last_viewed_at = 510619009; + LookModel model = 102848809; + // Is Public + bool public = 340099074; + // Public Slug + string public_slug = 351574145; + // Public Url + string public_url = 270773436; + // Query Id + string query_id = 279007282; + // Short Url + string short_url = 381379717; + FolderBase folder = 467441015; + // Folder Id + string folder_id = 271739380; + // Time that the Look was updated. + google.protobuf.Timestamp updated_at = 464589803; + // Number of times viewed in the Looker web UI + int64 view_count = 531861914; +} + + +message LookBasic { + // Operations the current user is able to perform on this object + map can = 108816; + // Id of content metadata + string content_metadata_id = 4111259; + // Unique Id + string id = 3205; + // Look Title + string title = 96607896; + // User Id + string user_id = 413336787; +} + + +message LookmlModel { + // Operations the current user is able to perform on this object + map can = 108816; + // Array of names of connections this model is allowed to use + repeated string allowed_db_connection_names = 350456152; + // Array of explores (if has_content) + repeated LookmlModelNavExplore explores = 343211776; + // Does this model declaration have have lookml content? + bool has_content = 529254405; + // UI-friendly name for this model + string label = 102846452; + // Name of the model. Also used as the unique identifier + string name = 3116757; + // Name of project containing the model + string project_name = 414886993; + // Is this model allowed to use all current and future connections + bool unlimited_db_connections = 413404803; +} + + +message LookmlModelExplore { + // Fully qualified explore name (model name plus explore name) + string id = 3205; + // Explore name + string name = 3116757; + // Description + string description = 375083905; + // Label + string label = 102846452; + // Explore title + string title = 96607896; + // Scopes + repeated string scopes = 452946431; + // Can Total + bool can_total = 379631994; + // Can Develop LookML + bool can_develop = 349133396; + // Can See LookML + bool can_see_lookml = 442342739; + // A URL linking to the definition of this explore in the LookML IDE. + string lookml_link = 454147565; + // Can Save + bool can_save = 167608180; + // Can Explain + bool can_explain = 306602916; + // Can pivot in the DB + bool can_pivot_in_db = 536547227; + // Can use subtotals + bool can_subtotal = 495966393; + // Has timezone support + bool has_timezone_support = 408767603; + // Cost estimates supported + bool supports_cost_estimate = 459772799; + // Connection name + string connection_name = 292836428; + // How nulls are sorted, possible values are "low", "high", "first" and "last" + string null_sort_treatment = 390349164; + // List of model source files + repeated string files = 109320951; + // Primary source_file file + string source_file = 354185832; + // Name of project + string project_name = 414886993; + // Name of model + string model_name = 275154383; + // Name of view + string view_name = 510594582; + // Is hidden + bool hidden = 524703253; + // A sql_table_name expression that defines what sql table the view/explore maps onto. Example: "prod_orders2 AS orders" in a view named orders. + string sql_table_name = 487960124; + // (DEPRECATED) Array of access filter field names + repeated string access_filter_fields = 342259097; + // Access filters + repeated LookmlModelExploreAccessFilter access_filters = 296481264; + // Aliases + repeated LookmlModelExploreAlias aliases = 496112863; + // Always filter + repeated LookmlModelExploreAlwaysFilter always_filter = 429286490; + // Conditionally filter + repeated LookmlModelExploreConditionallyFilter conditionally_filter = 278291982; + // Array of index fields + repeated string index_fields = 519627073; + // Sets + repeated LookmlModelExploreSet sets = 3540687; + // An array of arbitrary string tags provided in the model for this explore. + repeated string tags = 3528071; + // Errors + repeated LookmlModelExploreError errors = 446956773; + LookmlModelExploreFieldset fields = 453472492; + // Views joined into this explore + repeated LookmlModelExploreJoins joins = 109586377; + // Label used to group explores in the navigation menus + string group_label = 308675757; + // An array of items describing which custom measure types are supported for creating a custom measure 'based_on' each possible dimension type. + repeated LookmlModelExploreSupportedMeasureType supported_measure_types = 485636689; + // An array of joins that will always be included in the SQL for this explore, even if the user has not selected a field from the joined view. + repeated string always_join = 439273702; +} + + +message LookmlModelExploreAccessFilter { + // Field to be filtered + string field = 95669946; + // User attribute name + string user_attribute = 396252410; +} + + +message LookmlModelExploreAlias { + // Name + string name = 3116757; + // Value + string value = 96868081; +} + + +message LookmlModelExploreAlwaysFilter { + // Name + string name = 3116757; + // Value + string value = 96868081; +} + + +message LookmlModelExploreConditionallyFilter { + // Name + string name = 3116757; + // Value + string value = 96868081; +} + + +message LookmlModelExploreError { + // Error Message + string message = 452371230; + // Details + google.protobuf.Any details = 529834159; + // Error source location + string error_pos = 481055255; + // Is this a field error + bool field_error = 152347267; +} + + +message LookmlModelExploreField { + // The appropriate horizontal text alignment the values of this field should be displayed in. Valid values are: "left", "right". + Align align = 104760133; + // Whether it's possible to filter on this field. + bool can_filter = 418718630; + // Field category Valid values are: "parameter", "filter", "measure", "dimension". + Category category = 449734864; + // The default value that this field uses when filtering. Null if there is no default value. + string default_filter_value = 310754426; + // Description + string description = 375083905; + // Dimension group if this field is part of a dimension group. If not, this will be null. + string dimension_group = 390051315; + // An array enumerating all the possible values that this field can contain. When null, there is no limit to the set of possible values this field can contain. + repeated LookmlModelExploreFieldEnumeration enumerations = 277578548; + // An error message indicating a problem with the definition of this field. If there are no errors, this will be null. + string error = 108701384; + // A label creating a grouping of fields. All fields with this label should be presented together when displayed in a UI. + string field_group_label = 524667208; + // When presented in a field group via field_group_label, a shorter name of the field to be displayed in that context. + string field_group_variant = 432663856; + // The style of dimension fill that is possible for this field. Null if no dimension fill is possible. Valid values are: "enumeration", "range". + FillStyle fill_style = 459136074; + // An offset (in months) from the calendar start month to the fiscal start month defined in the LookML model this field belongs to. + int64 fiscal_month_offset = 480020859; + // Whether this field has a set of allowed_values specified in LookML. + bool has_allowed_values = 282386402; + // Whether this field should be hidden from the user interface. + bool hidden = 524703253; + // Whether this field is a filter. + bool is_filter = 298224404; + // Whether this field represents a fiscal time value. + bool is_fiscal = 339165593; + // Whether this field is of a type that represents a numeric value. + bool is_numeric = 354158564; + // Whether this field is of a type that represents a time value. + bool is_timeframe = 122615029; + // Whether this field can be time filtered. + bool can_time_filter = 446066363; + LookmlModelExploreFieldTimeInterval time_interval = 129899401; + // Fully-qualified human-readable label of the field. + string label = 102846452; + // The name of the parameter that will provide a parameterized label for this field, if available in the current context. + string label_from_parameter = 309078416; + // The human-readable label of the field, without the view label. + string label_short = 345179704; + // A URL linking to the definition of this field in the LookML IDE. + string lookml_link = 454147565; + LookmlModelExploreFieldMapLayer map_layer = 340672340; + // Whether this field is a measure. + bool measure = 369027784; + // Fully-qualified name of the field. + string name = 3116757; + // If yes, the field will not be localized with the user attribute number_format. Defaults to no + bool strict_value_format = 468791825; + // Whether this field is a parameter. + bool parameter = 347434709; + // Whether this field can be removed from a query. + bool permanent = 477498763; + // Whether or not the field represents a primary key. + bool primary_key = 253800418; + // The name of the project this field is defined in. + string project_name = 414886993; + // When true, it's not possible to re-sort this field's values without re-running the SQL query, due to database logic that affects the sort. + bool requires_refresh_on_sort = 128451113; + // The LookML scope this field belongs to. The scope is typically the field's view. + string scope = 96722068; + // Whether this field can be sorted. + bool sortable = 104858360; + // The path portion of source_file_path. + string source_file = 354185832; + // The fully-qualified path of the project file this field is defined in. + string source_file_path = 505031527; + // SQL expression as defined in the LookML model. The SQL syntax shown here is a representation intended for auditability, and is not neccessarily an exact match for what will ultimately be run in the database. It may contain special LookML syntax or annotations that are not valid SQL. This will be null if the current user does not have the see_lookml permission for the field's model. + string sql = 107406; + // An array of conditions and values that make up a SQL Case expression, as defined in the LookML model. The SQL syntax shown here is a representation intended for auditability, and is not neccessarily an exact match for what will ultimately be run in the database. It may contain special LookML syntax or annotations that are not valid SQL. This will be null if the current user does not have the see_lookml permission for the field's model. + repeated LookmlModelExploreFieldSqlCase sql_case = 446321671; + // Array of filter conditions defined for the measure in LookML. + repeated LookmlModelExploreFieldMeasureFilters filters = 487674337; + // The name of the dimension to base suggest queries from. + string suggest_dimension = 526000235; + // The name of the explore to base suggest queries from. + string suggest_explore = 322462956; + // Whether or not suggestions are possible for this field. + bool suggestable = 343759848; + // If available, a list of suggestions for this field. For most fields, a suggest query is a more appropriate way to get an up-to-date list of suggestions. Or use enumerations to list all the possible values. + repeated string suggestions = 530018100; + // An array of arbitrary string tags provided in the model for this field. + repeated string tags = 3528071; + // The LookML type of the field. + string type = 3120390; + // An array of user attribute types that are allowed to be used in filters on this field. Valid values are: "advanced_filter_string", "advanced_filter_number", "advanced_filter_datetime", "string", "number", "datetime", "relative_url", "yesno", "zipcode". + repeated UserAttributeFilterTypes user_attribute_filter_types = 60056361; + // If specified, the LookML value format string for formatting values of this field. + string value_format = 305564841; + // The name of the view this field belongs to. + string view = 3645563; + // The human-readable label of the view the field belongs to. + string view_label = 455664102; + // Whether this field was specified in "dynamic_fields" and is not part of the model. + bool dynamic = 389137567; + // The name of the starting day of the week. Valid values are: "monday", "tuesday", "wednesday", "thursday", "friday", "saturday", "sunday". + WeekStartDay week_start_day = 484849670; + // The number of times this field has been used in queries + int64 times_used = 451686026; + // The name of the view this field is defined in. This will be different than "view" when the view has been joined via a different name using the "from" parameter. + string original_view = 477738985; +} + + +message LookmlModelExploreFieldEnumeration { + // Label + string label = 102846452; + // Value + google.protobuf.Any value = 96868081; +} + + +message LookmlModelExploreFieldMapLayer { + // URL to the map layer resource. + string url = 107439; + // Name of the map layer, as defined in LookML. + string name = 3116757; + // Specifies the name of the TopoJSON object that the map layer references. If not specified, use the first object.. + string feature_key = 297733162; + // Selects which property from the TopoJSON data to plot against. TopoJSON supports arbitrary metadata for each region. When null, the first matching property should be used. + string property_key = 383099738; + // Which property from the TopoJSON data to use to label the region. When null, property_key should be used. + string property_label_key = 179738966; + // The preferred geographic projection of the map layer when displayed in a visualization that supports multiple geographic projections. + string projection = 521766443; + // Specifies the data format of the region information. Valid values are: "topojson", "vector_tile_region". + Format format = 440521963; + // Specifies the URL to a JSON file that defines the geographic extents of each region available in the map layer. This data is used to automatically center the map on the available data for visualization purposes. The JSON file must be a JSON object where the keys are the mapping value of the feature (as specified by property_key) and the values are arrays of four numbers representing the west longitude, south latitude, east longitude, and north latitude extents of the region. The object must include a key for every possible value of property_key. + string extents_json_url = 87688370; + // The minimum zoom level that the map layer may be displayed at, for visualizations that support zooming. + int64 max_zoom_level = 332362444; + // The maximum zoom level that the map layer may be displayed at, for visualizations that support zooming. + int64 min_zoom_level = 332364785; +} + + +message LookmlModelExploreFieldMeasureFilters { + // Filter field name + string field = 95669946; + // Filter condition value + string condition = 523284557; +} + + +message LookmlModelExploreFieldset { + // Array of dimensions + repeated LookmlModelExploreField dimensions = 101630381; + // Array of measures + repeated LookmlModelExploreField measures = 344645442; + // Array of filters + repeated LookmlModelExploreField filters = 487674337; + // Array of parameters + repeated LookmlModelExploreField parameters = 445252965; +} + + +message LookmlModelExploreFieldSqlCase { + // SQL Case label value + string value = 96868081; + // SQL Case condition expression + string condition = 523284557; +} + + +message LookmlModelExploreFieldTimeInterval { + // The type of time interval this field represents a grouping of. Valid values are: "day", "hour", "minute", "second", "millisecond", "microsecond", "week", "month", "quarter", "year". + Name name = 3116757; + // The number of intervals this field represents a grouping of. + int64 count = 110521423; +} + + +message LookmlModelExploreJoins { + // Name of this join (and name of the view to join) + string name = 3116757; + // Fields referenced by the join + repeated string dependent_fields = 357418681; + // Fields of the joined view to pull into this explore + repeated string fields = 453472492; + // Name of the dimension in this explore whose value is in the primary key of the joined view + string foreign_key = 306641939; + // Name of view to join + string from = 3357526; + // Specifies whether all queries must use an outer join + bool outer_only = 526280072; + // many_to_one, one_to_one, one_to_many, many_to_many + string relationship = 451220510; + // Names of joins that must always be included in SQL queries + repeated string required_joins = 286564983; + // SQL expression that produces a foreign key + string sql_foreign_key = 527009247; + // SQL ON expression describing the join condition + string sql_on = 520156152; + // SQL table name to join + string sql_table_name = 487960124; + // The join type: left_outer, full_outer, inner, or cross + string type = 3120390; + // Label to display in UI selectors + string view_label = 455664102; +} + + +message LookmlModelExploreSet { + // Name + string name = 3116757; + // Value set + repeated string value = 96868081; +} + + +message LookmlModelExploreSupportedMeasureType { + string dimension_type = 489160027; + repeated string measure_types = 371625804; +} + + +message LookmlModelNavExplore { + // Name of the explore + string name = 3116757; + // Description for the explore + string description = 375083905; + // Label for the explore + string label = 102846452; + // Is this explore marked as hidden + bool hidden = 524703253; + // Label used to group explores in the navigation menus + string group_label = 308675757; +} + + +message LookmlTest { + // Operations the current user is able to perform on this object + map can = 108816; + // Name of model containing this test. + string model_name = 275154383; + // Name of this test. + string name = 3116757; + // Name of the explore this test runs a query against + string explore_name = 403807127; + // The url parameters that can be used to reproduce this test's query on an explore. + string query_url_params = 279760195; + // Name of the LookML file containing this test. + string file = 3116036; + // Line number of this test in LookML. + int64 line = 3117964; +} + + +message LookmlTestResult { + // Operations the current user is able to perform on this object + map can = 108816; + // Name of model containing this test. + string model_name = 275154383; + // Name of this test. + string test_name = 510613594; + // Number of assertions in this test + int64 assertions_count = 343982168; + // Number of assertions passed in this test + int64 assertions_failed = 291077885; + // A list of any errors encountered by the test. + repeated ProjectError errors = 446956773; + // A list of any warnings encountered by the test. + repeated ProjectError warnings = 316777437; + // True if this test passsed without errors. + bool success = 480645727; +} + + +message LookModel { + // Model Id + string id = 3205; + // Model Label + string label = 102846452; +} + + +message LookWithDashboards { + // Operations the current user is able to perform on this object + map can = 108816; + // Id of content metadata + string content_metadata_id = 4111259; + // Unique Id + string id = 3205; + // Look Title + string title = 96607896; + // User Id + string user_id = 413336787; + // Content Favorite Id + string content_favorite_id = 363282886; + // Time that the Look was created. + google.protobuf.Timestamp created_at = 464589301; + // Whether or not a look is 'soft' deleted. + bool deleted = 389460870; + // Time that the Look was deleted. + google.protobuf.Timestamp deleted_at = 464526556; + // Id of User that deleted the look. + string deleter_id = 180578797; + // Description + string description = 375083905; + // Embed Url + string embed_url = 385173709; + // Excel File Url + string excel_file_url = 494304189; + // Number of times favorited + int64 favorite_count = 331565140; + // Google Spreadsheet Formula + string google_spreadsheet_formula = 494865843; + // Image Embed Url + string image_embed_url = 437097913; + // auto-run query when Look viewed + bool is_run_on_load = 487054535; + // Time that the Look was last accessed by any user + google.protobuf.Timestamp last_accessed_at = 324585985; + // Id of User that last updated the look. + string last_updater_id = 397418835; + // Time last viewed in the Looker web UI + google.protobuf.Timestamp last_viewed_at = 510619009; + LookModel model = 102848809; + // Is Public + bool public = 340099074; + // Public Slug + string public_slug = 351574145; + // Public Url + string public_url = 270773436; + // Query Id + string query_id = 279007282; + // Short Url + string short_url = 381379717; + FolderBase folder = 467441015; + // Folder Id + string folder_id = 271739380; + // Time that the Look was updated. + google.protobuf.Timestamp updated_at = 464589803; + // Number of times viewed in the Looker web UI + int64 view_count = 531861914; + // Dashboards + repeated DashboardBase dashboards = 301014335; +} + + +message LookWithQuery { + // Operations the current user is able to perform on this object + map can = 108816; + // Id of content metadata + string content_metadata_id = 4111259; + // Unique Id + string id = 3205; + // Look Title + string title = 96607896; + // User Id + string user_id = 413336787; + // Content Favorite Id + string content_favorite_id = 363282886; + // Time that the Look was created. + google.protobuf.Timestamp created_at = 464589301; + // Whether or not a look is 'soft' deleted. + bool deleted = 389460870; + // Time that the Look was deleted. + google.protobuf.Timestamp deleted_at = 464526556; + // Id of User that deleted the look. + string deleter_id = 180578797; + // Description + string description = 375083905; + // Embed Url + string embed_url = 385173709; + // Excel File Url + string excel_file_url = 494304189; + // Number of times favorited + int64 favorite_count = 331565140; + // Google Spreadsheet Formula + string google_spreadsheet_formula = 494865843; + // Image Embed Url + string image_embed_url = 437097913; + // auto-run query when Look viewed + bool is_run_on_load = 487054535; + // Time that the Look was last accessed by any user + google.protobuf.Timestamp last_accessed_at = 324585985; + // Id of User that last updated the look. + string last_updater_id = 397418835; + // Time last viewed in the Looker web UI + google.protobuf.Timestamp last_viewed_at = 510619009; + LookModel model = 102848809; + // Is Public + bool public = 340099074; + // Public Slug + string public_slug = 351574145; + // Public Url + string public_url = 270773436; + // Query Id + string query_id = 279007282; + // Short Url + string short_url = 381379717; + FolderBase folder = 467441015; + // Folder Id + string folder_id = 271739380; + // Time that the Look was updated. + google.protobuf.Timestamp updated_at = 464589803; + // Number of times viewed in the Looker web UI + int64 view_count = 531861914; + Query query = 115243016; + // Url + string url = 107439; +} + + +message Manifest { + // Operations the current user is able to perform on this object + map can = 108816; + // Manifest project name + string name = 3116757; + // Imports for a project + repeated ImportedProject imports = 470394564; + LocalizationSettings localization_settings = 454145053; +} + + +message MaterializePDT { + // The ID of the enqueued materialization task + string materialization_id = 347866311; + // Detailed response in text format + string resp_text = 491679969; +} + + +message MergeFields { + // Field name to map onto in the merged results + string field_name = 273359668; + // Field name from the source query + string source_field_name = 438453674; +} + + +message MergeQuery { + // Operations the current user is able to perform on this object + map can = 108816; + // Column Limit + string column_limit = 436244100; + // Dynamic Fields + string dynamic_fields = 405484086; + // Unique Id + string id = 3205; + // Pivots + repeated string pivots = 446031464; + // Unique to get results + string result_maker_id = 388730214; + // Sorts + repeated string sorts = 109773781; + // Source Queries defining the results to be merged. + repeated MergeQuerySourceQuery source_queries = 369604511; + // Total + bool total = 102745028; + // Visualization Config + map vis_config = 78818913; +} + + +message MergeQuerySourceQuery { + // An array defining which fields of the source query are mapped onto fields of the merge query + repeated MergeFields merge_fields = 523995608; + // Display name + string name = 3116757; + // Id of the query to merge + string query_id = 279007282; +} + + +message MobileSettings { + // Specifies whether the force authentication option is enabled for mobile + bool mobile_force_authentication = 463400349; + // Specifies whether mobile access for this instance is enabled. + bool mobile_app_integration = 275209422; +} + + +message Model { + string connection = 521736919; + string name = 3116757; + // Array of named value formats + repeated ModelNamedValueFormats value_formats = 301933401; +} + + +message ModelFieldSuggestions { + // List of suggestions + repeated string suggestions = 530018100; + // Error message + string error = 108701384; + // True if result came from the cache + bool from_cache = 526561203; + // True if this was a hit limit + bool hit_limit = 162570767; + // True if calcite was used + bool used_calcite_materialization = 347943687; +} + + +message ModelNamedValueFormats { + string format_string = 426314086; + string label = 102846452; + string name = 3116757; + bool strict_value_format = 468791825; +} + + +message ModelSet { + // Operations the current user is able to perform on this object + map can = 108816; + bool all_access = 499364607; + bool built_in = 35565638; + // Unique Id + string id = 3205; + repeated string models = 449883061; + // Name of ModelSet + string name = 3116757; + // Link to get this item + string url = 107439; +} + + +message ModelsNotValidated { + // Model name + string name = 3116757; + // Project file + string project_file_id = 457878570; +} + +// The type of time interval this field represents a grouping of. Valid values are: "day", "hour", "minute", "second", "millisecond", "microsecond", "week", "month", "quarter", "year". (Enum defined in LookmlModelExploreFieldTimeInterval) +enum Name { + _NAME_UNSET = 0; + NAME_DAY = 86142712; + NAME_HOUR = 411437844; + NAME_MINUTE = 457404348; + NAME_SECOND = 526634220; + NAME_MILLISECOND = 421554317; + NAME_MICROSECOND = 483528983; + NAME_WEEK = 317136106; + NAME_MONTH = 289256246; + NAME_QUARTER = 273650806; + NAME_YEAR = 279437667; +} + + +message OauthClientApp { + // Operations the current user is able to perform on this object + map can = 108816; + // The globally unique id of this application + string client_guid = 368914559; + // The uri with which this application will receive an auth code by browser redirect. + string redirect_uri = 379793333; + // The application's display name + string display_name = 322947240; + // A description of the application that will be displayed to users + string description = 375083905; + // When enabled is true, OAuth2 and API requests will be accepted from this app. When false, all requests from this app will be refused. + bool enabled = 387588912; + // If set, only Looker users who are members of this group can use this web app with Looker. If group_id is not set, any Looker user may use this app to access this Looker instance + string group_id = 287220091; + // All auth codes, access tokens, and refresh tokens issued for this application prior to this date-time for ALL USERS will be invalid. + google.protobuf.Timestamp tokens_invalid_before = 412773702; + // All users who have been activated to use this app + repeated UserPublic activated_users = 269658521; +} + + +message OIDCConfig { + // Operations the current user is able to perform on this object + map can = 108816; + // Allow alternate email-based login via '/login/email' for admins and for specified users with the 'login_special_email' permission. This option is useful as a fallback during ldap setup, if ldap config problems occur later, or if you need to support some users who are not in your ldap directory. Looker email/password logins are always disabled for regular users when ldap is enabled. + bool alternate_email_login_allowed = 415381803; + // OpenID Provider Audience + string audience = 422223006; + // Users will not be allowed to login at all unless a role for them is found in OIDC if set to true + bool auth_requires_role = 359416088; + // OpenID Provider Authorization Url + string authorization_endpoint = 3578117; + // (Write-Only) Array of ids of groups that will be applied to new users the first time they login via OIDC + repeated string default_new_user_group_ids = 417958896; + // (Read-only) Groups that will be applied to new users the first time they login via OIDC + repeated Group default_new_user_groups = 487231946; + // (Write-Only) Array of ids of roles that will be applied to new users the first time they login via OIDC + repeated string default_new_user_role_ids = 319694385; + // (Read-only) Roles that will be applied to new users the first time they login via OIDC + repeated Role default_new_user_roles = 478070081; + // Enable/Disable OIDC authentication for the server + bool enabled = 387588912; + // (Read-only) Array of mappings between OIDC Groups and Looker Roles + repeated OIDCGroupRead groups = 447792362; + // Name of user record attributes used to indicate groups. Used when 'groups_finder_type' is set to 'grouped_attribute_values' + string groups_attribute = 389492843; + // (Read/Write) Array of mappings between OIDC Groups and arrays of Looker Role ids + repeated OIDCGroupWrite groups_with_role_ids = 475375615; + // Relying Party Identifier (provided by OpenID Provider) + string identifier = 333214290; + // OpenID Provider Issuer + string issuer = 467184364; + // When this config was last modified + google.protobuf.Timestamp modified_at = 475018772; + // User id of user who last modified this config + string modified_by = 358701837; + // Merge first-time oidc login to existing user account by email addresses. When a user logs in for the first time via oidc this option will connect this user into their existing account by finding the account with a matching email address by testing the given types of credentials for existing users. Otherwise a new user account will be created for the user. This list (if provided) must be a comma separated list of string like 'email,ldap,google' + string new_user_migration_types = 331422246; + // Array of scopes to request. + repeated string scopes = 452946431; + // (Write-Only) Relying Party Secret (provided by OpenID Provider) + string secret = 438607800; + // Set user roles in Looker based on groups from OIDC + bool set_roles_from_groups = 390938262; + // Slug to identify configurations that are created in order to run a OIDC config test + string test_slug = 505468220; + // OpenID Provider Token Url + string token_endpoint = 534249150; + // Name of user record attributes used to indicate email address field + string user_attribute_map_email = 398964231; + // Name of user record attributes used to indicate first name + string user_attribute_map_first_name = 400740186; + // Name of user record attributes used to indicate last name + string user_attribute_map_last_name = 521517530; + // (Read-only) Array of mappings between OIDC User Attributes and Looker User Attributes + repeated OIDCUserAttributeRead user_attributes = 15243445; + // (Read/Write) Array of mappings between OIDC User Attributes and arrays of Looker User Attribute ids + repeated OIDCUserAttributeWrite user_attributes_with_ids = 332652790; + // OpenID Provider User Information Url + string userinfo_endpoint = 274918578; + // Allow OIDC auth'd users to be members of non-reflected Looker groups. If 'false', user will be removed from non-reflected groups on login. + bool allow_normal_group_membership = 215101880; + // OIDC auth'd users will inherit roles from non-reflected Looker groups. + bool allow_roles_from_normal_groups = 301477131; + // Allows roles to be directly assigned to OIDC auth'd users. + bool allow_direct_roles = 276166815; + // Link to get this item + string url = 107439; +} + + +message OIDCGroupRead { + // Unique Id + string id = 3205; + // Unique Id of group in Looker + string looker_group_id = 384327945; + // Name of group in Looker + string looker_group_name = 202505630; + // Name of group in OIDC + string name = 3116757; + // Looker Roles + repeated Role roles = 109321149; +} + + +message OIDCGroupWrite { + // Unique Id + string id = 3205; + // Unique Id of group in Looker + string looker_group_id = 384327945; + // Name of group in Looker + string looker_group_name = 202505630; + // Name of group in OIDC + string name = 3116757; + // Looker Role Ids + repeated string role_ids = 212692081; +} + + +message OIDCUserAttributeRead { + // Name of User Attribute in OIDC + string name = 3116757; + // Required to be in OIDC assertion for login to be allowed to succeed + bool required = 497611616; + // Looker User Attributes + repeated UserAttribute user_attributes = 15243445; +} + + +message OIDCUserAttributeWrite { + // Name of User Attribute in OIDC + string name = 3116757; + // Required to be in OIDC assertion for login to be allowed to succeed + bool required = 497611616; + // Looker User Attribute Ids + repeated string user_attribute_ids = 329947583; +} + + +message PasswordConfig { + // Operations the current user is able to perform on this object + map can = 108816; + // Minimum number of characters required for a new password. Must be between 7 and 100 + int64 min_length = 378159156; + // Require at least one numeric character + bool require_numeric = 438215331; + // Require at least one uppercase and one lowercase letter + bool require_upperlower = 330053382; + // Require at least one special character + bool require_special = 301715528; +} + + +message Permission { + // Operations the current user is able to perform on this object + map can = 108816; + // Permission symbol + string permission = 445562548; + // Dependency parent symbol + string parent = 434638453; + // Description + string description = 375083905; +} + + +message PermissionSet { + // Operations the current user is able to perform on this object + map can = 108816; + bool all_access = 499364607; + bool built_in = 35565638; + // Unique Id + string id = 3205; + // Name of PermissionSet + string name = 3116757; + repeated string permissions = 322818529; + // Link to get this item + string url = 107439; +} + +// Type of permission: "view" or "edit" Valid values are: "view", "edit". (Enum defined in ContentMetaGroupUser) +enum PermissionType { + _PERMISSION_TYPE_UNSET = 0; + PERMISSION_TYPE_VIEW = 168099814; + PERMISSION_TYPE_EDIT = 367654600; +} + + +message PrivatelabelConfiguration { + // Customer logo image. Expected base64 encoded data (write-only) + string logo_file = 302348988; + // Logo image url (read-only) + string logo_url = 390957090; + // Custom favicon image. Expected base64 encoded data (write-only) + string favicon_file = 348157260; + // Favicon image url (read-only) + string favicon_url = 432048593; + // Default page title + string default_title = 491557590; + // Boolean to toggle showing help menus + bool show_help_menu = 319857861; + // Boolean to toggle showing docs + bool show_docs = 430915951; + // Boolean to toggle showing email subscription options. + bool show_email_sub_options = 481532061; + // Boolean to toggle mentions of Looker in emails + bool allow_looker_mentions = 373852627; + // Boolean to toggle links to Looker in emails + bool allow_looker_links = 375199970; + // Allow subject line and email heading customization in customized emails” + bool custom_welcome_email_advanced = 119812016; + // Remove the word Looker from appearing in the account setup page + bool setup_mentions = 338894507; + // Remove Looker logo from Alerts + bool alerts_logo = 409573540; + // Remove Looker links from Alerts + bool alerts_links = 304998460; + // Remove Looker mentions in home folder page when you don’t have any items saved + bool folders_mentions = 496168113; +} + + +message Project { + // Operations the current user is able to perform on this object + map can = 108816; + // Project Id + string id = 3205; + // Project display name + string name = 3116757; + // If true the project is configured with a git repository + bool uses_git = 510301424; + // Git remote repository url + string git_remote_url = 409688477; + // Git username for HTTPS authentication. (For production only, if using user attributes.) + string git_username = 404090688; + // (Write-Only) Git password for HTTPS authentication. (For production only, if using user attributes.) + string git_password = 269826806; + // Git production branch name. Defaults to master. Supported only in Looker 21.0 and higher. + string git_production_branch_name = 384108369; + // If true, the project uses a git cookie for authentication. + bool use_git_cookie_auth = 517923168; + // User attribute name for username in per-user HTTPS authentication. + string git_username_user_attribute = 312364062; + // User attribute name for password in per-user HTTPS authentication. + string git_password_user_attribute = 517735463; + // Name of the git service provider + string git_service_name = 365333079; + // Port that HTTP(S) application server is running on (for PRs, file browsing, etc.) + int64 git_application_server_http_port = 529772882; + // Scheme that is running on application server (for PRs, file browsing, etc.) + string git_application_server_http_scheme = 330358350; + // (Write-Only) Optional secret token with which to authenticate requests to the webhook deploy endpoint. If not set, endpoint is unauthenticated. + string deploy_secret = 528157198; + // (Write-Only) When true, unsets the deploy secret to allow unauthenticated access to the webhook deploy endpoint. + bool unset_deploy_secret = 535705319; + // The git pull request policy for this project. Valid values are: "off", "links", "recommended", "required". + PullRequestMode pull_request_mode = 432598428; + // Validation policy: If true, the project must pass validation checks before project changes can be committed to the git repository + bool validation_required = 405012041; + // If true, advanced git release management is enabled for this project + bool git_release_mgmt_enabled = 350876702; + // Validation policy: If true, the project can be committed with warnings when `validation_required` is true. (`allow_warnings` does nothing if `validation_required` is false). + bool allow_warnings = 435037044; + // If true the project is an example project and cannot be modified + bool is_example = 458427965; + // Status of dependencies in your manifest & lockfile + string dependency_status = 440919897; +} + + +message ProjectError { + // A stable token that uniquely identifies this class of error, ignoring parameter values. Error message text may vary due to parameters or localization, but error codes do not. For example, a "File not found" error will have the same error code regardless of the filename in question or the user's display language + string code = 3108531; + // Severity: fatal, error, warning, info, success + string severity = 441317294; + // Error classification: syntax, deprecation, model_configuration, etc + string kind = 3088172; + // Error message which may contain information such as dashboard or model names that may be considered sensitive in some use cases. Avoid storing or sending this message outside of Looker + string message = 452371230; + // The field associated with this error + string field_name = 273359668; + // Name of the file containing this error + string file_path = 496692104; + // Line number in the file of this error + int64 line_number = 426831235; + // The model associated with this error + string model_id = 291401489; + // The explore associated with this error + string explore = 370461451; + // A link to Looker documentation about this error + string help_url = 390974231; + // Error parameters + map params = 449474371; + // A version of the error message that does not contain potentially sensitive information. Suitable for situations in which messages are stored or sent to consumers outside of Looker, such as external logs. Sanitized messages will display "(?)" where sensitive information would appear in the corresponding non-sanitized message + string sanitized_message = 336335153; +} + + +message ProjectFile { + // Operations the current user is able to perform on this object + map can = 108816; + // An opaque token uniquely identifying a file within a project. Avoid parsing or decomposing the text of this token. This token is stable within a Looker release but may change between Looker releases + string id = 3205; + // Path, file name, and extension of the file relative to the project root directory + string path = 3212859; + // Display name + string title = 96607896; + // File type: model, view, etc + string type = 3120390; + // The extension of the file: .view.lkml, .model.lkml, etc + string extension = 511296703; + // File mime type + string mime_type = 278104102; + // State of editability for the file. + bool editable = 104867364; + GitStatus git_status = 444546808; +} + + +message ProjectValidation { + // A list of project errors + repeated ProjectError errors = 446956773; + // A hash value computed from the project's current state + string project_digest = 273094778; + // A list of models which were not fully validated + repeated ModelsNotValidated models_not_validated = 296295883; + // Duration of project validation in seconds + float computation_time = 483498726; +} + + +message ProjectValidationCache { + // A list of project errors + repeated ProjectError errors = 446956773; + // A hash value computed from the project's current state + string project_digest = 273094778; + // A list of models which were not fully validated + repeated ModelsNotValidated models_not_validated = 296295883; + // Duration of project validation in seconds + float computation_time = 483498726; + // If true, the cached project validation results are no longer accurate because the project has changed since the cached results were calculated + bool stale = 96589977; +} + + +message ProjectWorkspace { + // Operations the current user is able to perform on this object + map can = 108816; + // The id of the project + string project_id = 471156823; + // The id of the local workspace containing the project files + string workspace_id = 456431629; + // The status of the local git directory + string git_status = 444546808; + // Git head revision name + string git_head = 341605228; + // Status of the dependencies in your project. Valid values are: "lock_optional", "lock_required", "lock_error", "install_none". + DependencyStatus dependency_status = 440919897; + GitBranch git_branch = 174392177; + // The lookml syntax used by all files in this project + string lookml_type = 307577421; +} + +// The git pull request policy for this project. Valid values are: "off", "links", "recommended", "required". (Enum defined in Project) +enum PullRequestMode { + _PULL_REQUEST_MODE_UNSET = 0; + PULL_REQUEST_MODE_OFF = 436551872; + PULL_REQUEST_MODE_LINKS = 521293372; + PULL_REQUEST_MODE_RECOMMENDED = 528921879; + PULL_REQUEST_MODE_REQUIRED = 325666539; +} + + +message Query { + // Operations the current user is able to perform on this object + map can = 108816; + // Unique Id + string id = 3205; + // Model + string model = 102848809; + // Explore Name + string view = 3645563; + // Fields + repeated string fields = 453472492; + // Pivots + repeated string pivots = 446031464; + // Fill Fields + repeated string fill_fields = 520489690; + // Filters + map filters = 487674337; + // Filter Expression + string filter_expression = 175166719; + // Sorting for the query results. Use the format `["view.field", ...]` to sort on fields in ascending order. Use the format `["view.field desc", ...]` to sort on fields in descending order. Use `["__UNSORTED__"]` (2 underscores before and after) to disable sorting entirely. Empty sorts `[]` will trigger a default sort. + repeated string sorts = 109773781; + // Limit + string limit = 110364603; + // Column Limit + string column_limit = 436244100; + // Total + bool total = 102745028; + // Raw Total + string row_total = 379629720; + // Fields on which to run subtotals + repeated string subtotals = 290419819; + // Visualization configuration properties. These properties are typically opaque and differ based on the type of visualization used. There is no specified set of allowed keys. The values can be any type supported by JSON. A "type" key with a string value is often present, and is used by Looker to determine which visualization to present. Visualizations ignore unknown vis_config properties. + map vis_config = 78818913; + // The filter_config represents the state of the filter UI on the explore page for a given query. When running a query via the Looker UI, this parameter takes precedence over "filters". When creating a query or modifying an existing query, "filter_config" should be set to null. Setting it to any other value could cause unexpected filtering behavior. The format should be considered opaque. + map filter_config = 305149514; + // Visible UI Sections + string visible_ui_sections = 364636603; + // Slug + string slug = 3184373; + // Dynamic Fields + string dynamic_fields = 405484086; + // Client Id: used to generate shortened explore URLs. If set by client, must be a unique 22 character alphanumeric string. Otherwise one will be generated. + string client_id = 206142577; + // Share Url + string share_url = 384846284; + // Expanded Share Url + string expanded_share_url = 379110377; + // Expanded Url + string url = 107439; + // Query Timezone + string query_timezone = 391786756; + // Has Table Calculations + bool has_table_calculations = 306982192; +} + + +message QueryTask { + // Operations the current user is able to perform on this object + map can = 108816; + // Unique Id + string id = 3205; + // Id of query + string query_id = 279007282; + Query query = 115243016; + // whether or not to generate links in the query response. + bool generate_links = 503783544; + // Use production models to run query (even is user is in dev mode). + bool force_production = 426087380; + // Prefix to use for drill links. + string path_prefix = 400540043; + // Whether or not to use the cache + bool cache = 96472130; + // Whether or not to run table calculations on the server + bool server_table_calcs = 437532812; + // Retrieve any results from cache even if the results have expired. + bool cache_only = 520313387; + // cache key used to cache query. + string cache_key = 366792847; + // Status of query task. + string status = 445505145; + // Source of query task. + string source = 327120574; + // Runtime of prior queries. + float runtime = 407574090; + // Rebuild PDTS used in query. + bool rebuild_pdts = 367772772; + // Source of the results of the query. + string result_source = 436913583; + // Id of look associated with query. + string look_id = 413287022; + // Id of dashboard associated with query. + string dashboard_id = 311679049; + // The data format of the query results. + string result_format = 296518982; +} + + +message RenderTask { + // Operations the current user is able to perform on this object + map can = 108816; + // Date/Time render task was created + string created_at = 464589301; + // Filter values to apply to the dashboard queries, in URL query format + string dashboard_filters = 470460328; + // Id of dashboard to render + string dashboard_id = 311679049; + // Dashboard layout style: single_column or tiled + string dashboard_style = 515014529; + // Date/Time render task was completed + string finalized_at = 419405293; + // Output height in pixels. Flowed layouts may ignore this value. + int64 height = 437383491; + // Id of this render task + string id = 3205; + // Id of look to render + string look_id = 413287022; + // Id of lookml dashboard to render + string lookml_dashboard_id = 351003409; + // Id of query to render + string query_id = 279007282; + // Id of dashboard element to render: UDD dashboard element would be numeric and LookML dashboard element would be model_name::dashboard_title::lookml_link_id + string dashboard_element_id = 426786746; + // Number of seconds elapsed running queries + double query_runtime = 493944839; + // Number of seconds elapsed rendering data + double render_runtime = 333016147; + // Output format: pdf, png, or jpg + string result_format = 296518982; + // Total seconds elapsed for render task + double runtime = 407574090; + // Render task status: enqueued_for_query, querying, enqueued_for_render, rendering, success, failure + string status = 445505145; + // Additional information about the current status + string status_detail = 412551311; + // The user account permissions in which the render task will execute + string user_id = 413336787; + // Output width in pixels + int64 width = 99601414; +} + + +message RepositoryCredential { + // Operations the current user is able to perform on this object + map can = 108816; + // Unique Id + string id = 3205; + // Root project Id + string root_project_id = 487670047; + // Git remote repository url + string remote_url = 474030390; + // Git username for HTTPS authentication. + string git_username = 404090688; + // (Write-Only) Git password for HTTPS authentication. + string git_password = 269826806; + // Public deploy key for SSH authentication. + string ssh_public_key = 525275256; + // Whether the credentials have been configured for the Git Repository. + bool is_configured = 269688556; +} + +// Desired async query result format. Valid values are: "inline_json", "json", "json_detail", "json_fe", "csv", "html", "md", "txt", "xlsx", "gsxml". (Enum defined in CreateQueryTask) +enum ResultFormat { + _RESULT_FORMAT_UNSET = 0; + RESULT_FORMAT_INLINE_JSON = 495803702; + RESULT_FORMAT_JSON = 348577084; + RESULT_FORMAT_JSON_DETAIL = 374102519; + RESULT_FORMAT_JSON_FE = 406069588; + RESULT_FORMAT_CSV = 388422880; + RESULT_FORMAT_HTML = 362790787; + RESULT_FORMAT_MD = 403755768; + RESULT_FORMAT_TXT = 328900246; + RESULT_FORMAT_XLSX = 41451681; + RESULT_FORMAT_GSXML = 425806297; +} + + +message ResultMakerFilterables { + // The model this filterable comes from (used for field suggestions). + string model = 102848809; + // The view this filterable comes from (used for field suggestions). + string view = 3645563; + // The name of the filterable thing (Query or Merged Results). + string name = 3116757; + // array of dashboard_filter_name: and field: objects. + repeated ResultMakerFilterablesListen listen = 524457715; +} + + +message ResultMakerFilterablesListen { + // The name of a dashboard filter to listen to. + string dashboard_filter_name = 414233337; + // The name of the field in the filterable to filter with the value of the dashboard filter. + string field = 95669946; +} + + +message ResultMakerWithIdVisConfigAndDynamicFields { + // Unique Id. + string id = 3205; + // JSON string of dynamic field information. + string dynamic_fields = 405484086; + // array of items that can be filtered and information about them. + repeated ResultMakerFilterables filterables = 437559144; + // Sorts of the constituent Look, Query, or Merge Query + repeated string sorts = 109773781; + // ID of merge result if this is a merge_result. + string merge_result_id = 3223242; + // Total of the constituent Look, Query, or Merge Query + bool total = 102745028; + // ID of query if this is a query. + string query_id = 279007282; + // ID of SQL Query if this is a SQL Runner Query + string sql_query_id = 277038360; + Query query = 115243016; + // Vis config of the constituent Query, or Merge Query. + map vis_config = 78818913; +} + + +message Role { + // Operations the current user is able to perform on this object + map can = 108816; + // Unique Id + string id = 3205; + // Name of Role + string name = 3116757; + PermissionSet permission_set = 460987964; + // (Write-Only) Id of permission set + string permission_set_id = 239948280; + ModelSet model_set = 442444109; + // (Write-Only) Id of model set + string model_set_id = 339578852; + // Link to get this item + string url = 107439; + // Link to get list of users with this role + string users_url = 381612914; +} + + +message RoleSearch { + // Operations the current user is able to perform on this object + map can = 108816; + // Unique Id + string id = 3205; + // Name of Role + string name = 3116757; + PermissionSet permission_set = 460987964; + // (Write-Only) Id of permission set + string permission_set_id = 239948280; + ModelSet model_set = 442444109; + // (Write-Only) Id of model set + string model_set_id = 339578852; + // Count of users with this role + int64 user_count = 531824753; + // Link to get this item + string url = 107439; + // Link to get list of users with this role + string users_url = 381612914; +} + + +message RunningQueries { + // Operations the current user is able to perform on this object + map can = 108816; + // Unique Id + string id = 3205; + UserPublic user = 3496917; + Query query = 115243016; + SqlQuery sql_query = 58722007; + LookBasic look = 3297857; + // Date/Time Query was initiated + string created_at = 464589301; + // Date/Time Query was completed + string completed_at = 533495865; + // Query Id + string query_id = 279007282; + // Source (look, dashboard, queryrunner, explore, etc.) + string source = 327120574; + // Node Id + string node_id = 413239694; + // Slug + string slug = 3184373; + // ID of a Query Task + string query_task_id = 297690007; + // Cache Key + string cache_key = 366792847; + // Connection + string connection_name = 292836428; + // Dialect + string dialect = 373219958; + // Connection ID + string connection_id = 394619036; + // Additional Information(Error message or verbose status) + string message = 452371230; + // Status description + string status = 445505145; + // Number of seconds elapsed running the Query + double runtime = 407574090; + // SQL text of the query as run + string sql = 107406; +} + + +message SamlConfig { + // Operations the current user is able to perform on this object + map can = 108816; + // Enable/Disable Saml authentication for the server + bool enabled = 387588912; + // Identity Provider Certificate (provided by IdP) + string idp_cert = 346163572; + // Identity Provider Url (provided by IdP) + string idp_url = 440865893; + // Identity Provider Issuer (provided by IdP) + string idp_issuer = 332335031; + // Identity Provider Audience (set in IdP config). Optional in Looker. Set this only if you want Looker to validate the audience value returned by the IdP. + string idp_audience = 323325369; + // Count of seconds of clock drift to allow when validating timestamps of assertions. + int64 allowed_clock_drift = 535374760; + // Name of user record attributes used to indicate email address field + string user_attribute_map_email = 398964231; + // Name of user record attributes used to indicate first name + string user_attribute_map_first_name = 400740186; + // Name of user record attributes used to indicate last name + string user_attribute_map_last_name = 521517530; + // Merge first-time saml login to existing user account by email addresses. When a user logs in for the first time via saml this option will connect this user into their existing account by finding the account with a matching email address by testing the given types of credentials for existing users. Otherwise a new user account will be created for the user. This list (if provided) must be a comma separated list of string like 'email,ldap,google' + string new_user_migration_types = 331422246; + // Allow alternate email-based login via '/login/email' for admins and for specified users with the 'login_special_email' permission. This option is useful as a fallback during ldap setup, if ldap config problems occur later, or if you need to support some users who are not in your ldap directory. Looker email/password logins are always disabled for regular users when ldap is enabled. + bool alternate_email_login_allowed = 415381803; + // Slug to identify configurations that are created in order to run a Saml config test + string test_slug = 505468220; + // When this config was last modified + string modified_at = 475018772; + // User id of user who last modified this config + string modified_by = 358701837; + // (Read-only) Roles that will be applied to new users the first time they login via Saml + repeated Role default_new_user_roles = 478070081; + // (Read-only) Groups that will be applied to new users the first time they login via Saml + repeated Group default_new_user_groups = 487231946; + // (Write-Only) Array of ids of roles that will be applied to new users the first time they login via Saml + repeated string default_new_user_role_ids = 319694385; + // (Write-Only) Array of ids of groups that will be applied to new users the first time they login via Saml + repeated string default_new_user_group_ids = 417958896; + // Set user roles in Looker based on groups from Saml + bool set_roles_from_groups = 390938262; + // Name of user record attributes used to indicate groups. Used when 'groups_finder_type' is set to 'grouped_attribute_values' + string groups_attribute = 389492843; + // (Read-only) Array of mappings between Saml Groups and Looker Roles + repeated SamlGroupRead groups = 447792362; + // (Read/Write) Array of mappings between Saml Groups and arrays of Looker Role ids + repeated SamlGroupWrite groups_with_role_ids = 475375615; + // Users will not be allowed to login at all unless a role for them is found in Saml if set to true + bool auth_requires_role = 359416088; + // (Read-only) Array of mappings between Saml User Attributes and Looker User Attributes + repeated SamlUserAttributeRead user_attributes = 15243445; + // (Read/Write) Array of mappings between Saml User Attributes and arrays of Looker User Attribute ids + repeated SamlUserAttributeWrite user_attributes_with_ids = 332652790; + // Identifier for a strategy for how Looker will find groups in the SAML response. One of ['grouped_attribute_values', 'individual_attributes'] + string groups_finder_type = 21587272; + // Value for group attribute used to indicate membership. Used when 'groups_finder_type' is set to 'individual_attributes' + string groups_member_value = 527664343; + // Bypass the login page when user authentication is required. Redirect to IdP immediately instead. + bool bypass_login_page = 327461774; + // Allow SAML auth'd users to be members of non-reflected Looker groups. If 'false', user will be removed from non-reflected groups on login. + bool allow_normal_group_membership = 215101880; + // SAML auth'd users will inherit roles from non-reflected Looker groups. + bool allow_roles_from_normal_groups = 301477131; + // Allows roles to be directly assigned to SAML auth'd users. + bool allow_direct_roles = 276166815; + // Link to get this item + string url = 107439; +} + + +message SamlGroupRead { + // Unique Id + string id = 3205; + // Unique Id of group in Looker + string looker_group_id = 384327945; + // Name of group in Looker + string looker_group_name = 202505630; + // Name of group in Saml + string name = 3116757; + // Looker Roles + repeated Role roles = 109321149; + // Link to saml config + string url = 107439; +} + + +message SamlGroupWrite { + // Unique Id + string id = 3205; + // Unique Id of group in Looker + string looker_group_id = 384327945; + // Name of group in Looker + string looker_group_name = 202505630; + // Name of group in Saml + string name = 3116757; + // Looker Role Ids + repeated string role_ids = 212692081; + // Link to saml config + string url = 107439; +} + + +message SamlMetadataParseResult { + // Operations the current user is able to perform on this object + map can = 108816; + // Identify Provider Issuer + string idp_issuer = 332335031; + // Identify Provider Url + string idp_url = 440865893; + // Identify Provider Certificate + string idp_cert = 346163572; +} + + +message SamlUserAttributeRead { + // Name of User Attribute in Saml + string name = 3116757; + // Required to be in Saml assertion for login to be allowed to succeed + bool required = 497611616; + // Looker User Attributes + repeated UserAttribute user_attributes = 15243445; + // Link to saml config + string url = 107439; +} + + +message SamlUserAttributeWrite { + // Name of User Attribute in Saml + string name = 3116757; + // Required to be in Saml assertion for login to be allowed to succeed + bool required = 497611616; + // Looker User Attribute Ids + repeated string user_attribute_ids = 329947583; + // Link to saml config + string url = 107439; +} + + +message ScheduledPlan { + // Name of this scheduled plan + string name = 3116757; + // User Id which owns this scheduled plan + string user_id = 413336787; + // Whether schedule is run as recipient (only applicable for email recipients) + bool run_as_recipient = 343878752; + // Whether the ScheduledPlan is enabled + bool enabled = 387588912; + // Id of a look + string look_id = 413287022; + // Id of a dashboard + string dashboard_id = 311679049; + // Id of a LookML dashboard + string lookml_dashboard_id = 351003409; + // Query string to run look or dashboard with + string filters_string = 457599877; + // (DEPRECATED) Alias for filters_string field + string dashboard_filters = 470460328; + // Delivery should occur if running the dashboard or look returns results + bool require_results = 367571767; + // Delivery should occur if the dashboard look does not return results + bool require_no_results = 342203479; + // Delivery should occur if data have changed since the last run + bool require_change = 451184898; + // Will run an unlimited query and send all results. + bool send_all_results = 332106111; + // Vixie-Style crontab specification when to run + string crontab = 331409081; + // Name of a datagroup; if specified will run when datagroup triggered (can't be used with cron string) + string datagroup = 269453013; + // Timezone for interpreting the specified crontab (default is Looker instance timezone) + string timezone = 516241033; + // Query id + string query_id = 279007282; + // Scheduled plan destinations + repeated ScheduledPlanDestination scheduled_plan_destination = 274258130; + // Whether the plan in question should only be run once (usually for testing) + bool run_once = 426696469; + // Whether links back to Looker should be included in this ScheduledPlan + bool include_links = 419825745; + // The size of paper the PDF should be formatted to fit. Valid values are: "letter", "legal", "tabloid", "a0", "a1", "a2", "a3", "a4", "a5". + string pdf_paper_size = 317503911; + // Whether the PDF should be formatted for landscape orientation + bool pdf_landscape = 272762770; + // Whether this schedule is in an embed context or not + bool embed = 95458649; + // Color scheme of the dashboard if applicable + string color_theme = 350454445; + // Whether or not to expand table vis to full length + bool long_tables = 450392146; + // The pixel width at which we render the inline table visualizations + int64 inline_table_width = 380760423; + // Unique Id + string id = 3205; + // Date and time when ScheduledPlan was created + google.protobuf.Timestamp created_at = 464589301; + // Date and time when ScheduledPlan was last updated + google.protobuf.Timestamp updated_at = 464589803; + // Title + string title = 96607896; + UserPublic user = 3496917; + // When the ScheduledPlan will next run (null if running once) + google.protobuf.Timestamp next_run_at = 382498563; + // When the ScheduledPlan was last run + google.protobuf.Timestamp last_run_at = 382499796; + // Operations the current user is able to perform on this object + map can = 108816; +} + + +message ScheduledPlanDestination { + // Unique Id + string id = 3205; + // Id of a scheduled plan you own + string scheduled_plan_id = 303270096; + // The data format to send to the given destination. Supported formats vary by destination, but include: "txt", "csv", "inline_json", "json", "json_detail", "xlsx", "html", "wysiwyg_pdf", "assembled_pdf", "wysiwyg_png" + string format = 440521963; + // Are values formatted? (containing currency symbols, digit separators, etc. + bool apply_formatting = 423291115; + // Whether visualization options are applied to the results. + bool apply_vis = 273430419; + // Address for recipient. For email e.g. 'user@example.com'. For webhooks e.g. 'https://domain/path'. For Amazon S3 e.g. 's3://bucket-name/path/'. For SFTP e.g. 'sftp://host-name/path/'. + string address = 480533907; + // Whether the recipient is a Looker user on the current instance (only applicable for email recipients) + bool looker_recipient = 356167155; + // Type of the address ('email', 'webhook', 's3', or 'sftp') + string type = 3120390; + // JSON object containing parameters for external scheduling. For Amazon S3, this requires keys and values for access_key_id and region. For SFTP, this requires a key and value for username. + string parameters = 445252965; + // (Write-Only) JSON object containing secret parameters for external scheduling. For Amazon S3, this requires a key and value for secret_access_key. For SFTP, this requires a key and value for password. + string secret_parameters = 480072537; + // Optional message to be included in scheduled emails + string message = 452371230; +} + + +message Schema { + // Schema name + string name = 3116757; + // True if this is the default schema + bool is_default = 499524181; +} + + +message SchemaColumn { + // Schema item name + string name = 3116757; + // Full name of item + string sql_escaped_name = 409458356; + // Name of schema + string schema_name = 274052203; + // SQL dialect data type + string data_type_database = 460336533; + // Data type + string data_type = 278132276; + // Looker data type + string data_type_looker = 287245289; + // SQL data type + string description = 375083905; + // Column data size + int64 column_size = 114230410; + // SQL Runner snippets for this connection + repeated Snippet snippets = 357339282; +} + + +message SchemaColumns { + // Schema item name + string name = 3116757; + // Full name of item + string sql_escaped_name = 409458356; + // Name of schema + string schema_name = 274052203; + // Columns for this schema + repeated SchemaColumn columns = 514448952; +} + + +message SchemaTable { + // Schema item name + string name = 3116757; + // Full name of item + string sql_escaped_name = 409458356; + // Name of schema + string schema_name = 274052203; + // Number of data rows + int64 rows = 3543879; + // External reference??? + string external = 400972202; + // SQL Runner snippets for connection + repeated Snippet snippets = 357339282; +} + + +message SchemaTables { + // Schema name + string name = 3116757; + // True if this is the default schema + bool is_default = 499524181; + // Tables for this schema + repeated SchemaTable tables = 453012290; + // True if the table limit was hit while retrieving tables in this schema + bool table_limit_hit = 515000759; +} + + +message Session { + // Operations the current user is able to perform on this object + map can = 108816; + // Unique Id + string id = 3205; + // IP address of user when this session was initiated + string ip_address = 459602151; + // User's browser type + string browser = 274402282; + // User's Operating System + string operating_system = 248330509; + // City component of user location (derived from IP address) + string city = 3719541; + // State component of user location (derived from IP address) + string state = 96828305; + // Country component of user location (derived from IP address) + string country = 453479829; + // Type of credentials used for logging in this session + string credentials_type = 484217920; + // Time when this session was last extended by the user + string extended_at = 407088220; + // Number of times this session was extended + int64 extended_count = 496509370; + // Actual user in the case when this session represents one user sudo'ing as another + string sudo_user_id = 324097672; + // Time when this session was initiated + string created_at = 464589301; + // Time when this session will expire + string expires_at = 503123230; + // Link to get this item + string url = 107439; +} + + +message SessionConfig { + // Operations the current user is able to perform on this object + map can = 108816; + // Allow users to have persistent sessions when they login + bool allow_persistent_sessions = 399922247; + // Number of minutes for user sessions. Must be between 5 and 43200 + int64 session_minutes = 444808605; + // Allow users to have an unbounded number of concurrent sessions (otherwise, users will be limited to only one session at a time). + bool unlimited_sessions_per_user = 298548380; + // Enforce session logout for sessions that are inactive for 15 minutes. + bool use_inactivity_based_logout = 299884770; + // Track location of session when user logs in. + bool track_session_location = 330330067; +} + + +message Setting { + // Toggle extension framework on or off + bool extension_framework_enabled = 350281536; + // Toggle marketplace auto install on or off. Note that auto install only runs if marketplace is enabled. + bool marketplace_auto_install_enabled = 407899176; + // Toggle marketplace on or off + bool marketplace_enabled = 518632083; + PrivatelabelConfiguration privatelabel_configuration = 150815432; + CustomWelcomeEmail custom_welcome_email = 199760943; + // Toggle onboarding on or off + bool onboarding_enabled = 456555871; +} + + +message SmtpNodeStatus { + // SMTP status of node + bool is_valid = 71120505; + // Error message for node + string message = 452371230; + // Host name of node + string hostname = 383011446; +} + + +message SmtpSettings { + // SMTP Server url + string address = 480533907; + // From e-mail address + string from = 3357526; + // User name + string user_name = 510631744; + // Password + string password = 425146747; + // SMTP Server's port + int64 port = 3568863; + // Is TLS encryption enabled? + bool enable_starttls_auto = 28768633; + // TLS version selected Valid values are: "TLSv1_1", "SSLv23", "TLSv1_2". + SslVersion ssl_version = 394771638; +} + + +message SmtpStatus { + // Overall SMTP status of cluster + bool is_valid = 71120505; + // Total number of nodes in cluster + int64 node_count = 531727659; + // array of each node's status containing is_valid, message, hostname + repeated SmtpNodeStatus node_status = 355610968; +} + + +message Snippet { + // Name of the snippet + string name = 3116757; + // Label of the snippet + string label = 102846452; + // SQL text of the snippet + string sql = 107406; +} + + +message SqlQuery { + // Operations the current user is able to perform on this object + map can = 108816; + // The identifier of the SQL query + string slug = 3184373; + // Number of seconds this query took to run the most recent time it was run + float last_runtime = 434308092; + // Number of times this query has been run + int64 run_count = 519879730; + // Maximum number of rows this query will display on the SQL Runner page + int64 browser_limit = 382400914; + // SQL query text + string sql = 107406; + // The most recent time this query was run + string last_run_at = 382499796; + DBConnectionBase connection = 521736919; + // Model name this query uses + string model_name = 275154383; + UserPublic creator = 346039787; + // Explore page URL for this SQL query + string explore_url = 466829359; + // Should this query be rendered as plain text + bool plaintext = 488273170; + // Visualization configuration properties. These properties are typically opaque and differ based on the type of visualization used. There is no specified set of allowed keys. The values can be any type supported by JSON. A "type" key with a string value is often present, and is used by Looker to determine which visualization to present. Visualizations ignore unknown vis_config properties. + map vis_config = 78818913; + // ID of the ResultMakerLookup entry. + string result_maker_id = 388730214; +} + + +message SqlQueryCreate { + // Name of the db connection on which to run this query + string connection_name = 292836428; + // (DEPRECATED) Use `connection_name` instead + string connection_id = 394619036; + // Name of LookML Model (this or `connection_id` required) + string model_name = 275154383; + // SQL query + string sql = 107406; + // Visualization configuration properties. These properties are typically opaque and differ based on the type of visualization used. There is no specified set of allowed keys. The values can be any type supported by JSON. A "type" key with a string value is often present, and is used by Looker to determine which visualization to present. Visualizations ignore unknown vis_config properties. + map vis_config = 78818913; +} + + +message SshPublicKey { + // The SSH public key created for this instance + string public_key = 330596266; +} + + +message SshServer { + // A unique id used to identify this SSH Server + string ssh_server_id = 20333920; + // The name to identify this SSH Server + string ssh_server_name = 122116528; + // The hostname or ip address of the SSH Server + string ssh_server_host = 422692139; + // The port to connect to on the SSH Server + int64 ssh_server_port = 532989181; + // The username used to connect to the SSH Server + string ssh_server_user = 490745428; + // The md5 fingerprint used to identify the SSH Server + string finger_print = 488236429; + // The SHA fingerprint used to identify the SSH Server + string sha_finger_print = 381910309; + // The SSH public key created for this instance + string public_key = 330596266; + // The current connection status to this SSH Server + string status = 445505145; +} + + +message SshTunnel { + // Unique ID for the tunnel + string tunnel_id = 443214254; + // SSH Server ID + string ssh_server_id = 20333920; + // SSH Server name + string ssh_server_name = 122116528; + // SSH Server Hostname or IP Address + string ssh_server_host = 422692139; + // SSH Server port + int64 ssh_server_port = 532989181; + // Username used to connect to the SSH Server + string ssh_server_user = 490745428; + // Time of last connect attempt + string last_attempt = 334346802; + // Localhost Port used by the Looker instance to connect to the remote DB + int64 local_host_port = 385810830; + // Hostname or IP Address of the Database Server + string database_host = 495546218; + // Port that the Database Server is listening on + int64 database_port = 400671025; + // Current connection status for this Tunnel + string status = 445505145; +} + +// TLS version selected Valid values are: "TLSv1_1", "SSLv23", "TLSv1_2". (Enum defined in SmtpSettings) +enum SslVersion { + _SSL_VERSION_UNSET = 0; + SSL_VERSION_TLSV1_1 = 143875506; + SSL_VERSION_SSLV23 = 353074661; + SSL_VERSION_TLSV1_2 = 494173688; +} + + +message SupportAccessAddEntries { + // An array of emails to add to the Allowlist + repeated string emails = 449824955; + // Reason for adding emails to the Allowlist + string reason = 519863714; +} + + +message SupportAccessAllowlistEntry { + // Unique ID + string id = 3205; + // Email address + string email = 102965020; + // Full name of allowlisted user + string full_name = 510674737; + // Reason the Email is included in the Allowlist + string reason = 519863714; + // Date the Email was added to the Allowlist + google.protobuf.Timestamp created_date = 458292805; +} + + +message SupportAccessEnable { + // Duration Support Access will remain enabled + int64 duration_in_seconds = 33459152; +} + + +message SupportAccessStatus { + // Whether or not Support Access is open + bool open = 3377654; + // Time that Support Access will expire + google.protobuf.Timestamp open_until = 507909053; +} + +// A list of action types the integration supports. Valid values are: "cell", "query", "dashboard", "none". (Enum defined in Integration) +enum SupportedActionTypes { + _SUPPORTED_ACTION_TYPES_UNSET = 0; + SUPPORTED_ACTION_TYPES_CELL = 373174528; + SUPPORTED_ACTION_TYPES_QUERY = 126018058; + SUPPORTED_ACTION_TYPES_DASHBOARD = 289823947; + SUPPORTED_ACTION_TYPES_NONE = 378292581; +} + +// A list of all the download mechanisms the integration supports. The order of values is not significant: Looker will select the most appropriate supported download mechanism for a given query. The integration must ensure it can handle any of the mechanisms it claims to support. If unspecified, this defaults to all download setting values. Valid values are: "push", "url". (Enum defined in Integration) +enum SupportedDownloadSettings { + _SUPPORTED_DOWNLOAD_SETTINGS_UNSET = 0; + SUPPORTED_DOWNLOAD_SETTINGS_PUSH = 314540168; + SUPPORTED_DOWNLOAD_SETTINGS_URL = 477265651; +} + +// A list of data formats the integration supports. If unspecified, the default is all data formats. Valid values are: "txt", "csv", "inline_json", "json", "json_label", "json_detail", "json_detail_lite_stream", "xlsx", "html", "wysiwyg_pdf", "assembled_pdf", "wysiwyg_png", "csv_zip". (Enum defined in Integration) +enum SupportedFormats { + _SUPPORTED_FORMATS_UNSET = 0; + SUPPORTED_FORMATS_TXT = 483871129; + SUPPORTED_FORMATS_CSV = 500614588; + SUPPORTED_FORMATS_INLINE_JSON = 501250605; + SUPPORTED_FORMATS_JSON = 286983607; + SUPPORTED_FORMATS_JSON_LABEL = 262463983; + SUPPORTED_FORMATS_JSON_DETAIL = 308153260; + SUPPORTED_FORMATS_JSON_DETAIL_LITE_STREAM = 258775046; + SUPPORTED_FORMATS_XLSX = 420840787; + SUPPORTED_FORMATS_HTML = 509371455; + SUPPORTED_FORMATS_WYSIWYG_PDF = 511939053; + SUPPORTED_FORMATS_ASSEMBLED_PDF = 424032340; + SUPPORTED_FORMATS_WYSIWYG_PNG = 392947331; + SUPPORTED_FORMATS_CSV_ZIP = 386055763; +} + +// A list of formatting options the integration supports. If unspecified, defaults to all formats. Valid values are: "formatted", "unformatted". (Enum defined in Integration) +enum SupportedFormattings { + _SUPPORTED_FORMATTINGS_UNSET = 0; + SUPPORTED_FORMATTINGS_FORMATTED = 489436190; + SUPPORTED_FORMATTINGS_UNFORMATTED = 292502784; +} + +// A list of visualization formatting options the integration supports. If unspecified, defaults to all formats. Valid values are: "apply", "noapply". (Enum defined in Integration) +enum SupportedVisualizationFormattings { + _SUPPORTED_VISUALIZATION_FORMATTINGS_UNSET = 0; + SUPPORTED_VISUALIZATION_FORMATTINGS_APPLY = 536243731; + SUPPORTED_VISUALIZATION_FORMATTINGS_NOAPPLY = 369419100; +} + + +message Theme { + // Operations the current user is able to perform on this object + map can = 108816; + // Timestamp for when this theme becomes active. Null=always + google.protobuf.Timestamp begin_at = 489666644; + // Timestamp for when this theme expires. Null=never + google.protobuf.Timestamp end_at = 440737243; + // Unique Id + string id = 3205; + // Name of theme. Can only be alphanumeric and underscores. + string name = 3116757; + ThemeSettings settings = 316732240; +} + + +message ThemeSettings { + // Default background color + string background_color = 351188052; + // Base font size for scaling fonts + string base_font_size = 304368803; + // Optional. ID of color collection to use with the theme. Use an empty string for none. + string color_collection_id = 73484736; + // Default font color + string font_color = 521862868; + // Primary font family + string font_family = 476110122; + // Source specification for font + string font_source = 471962930; + // Info button color + string info_button_color = 359638350; + // Primary button color + string primary_button_color = 143819731; + // Toggle to show filters. Defaults to true. + bool show_filters_bar = 70091731; + // Toggle to show the title. Defaults to true. + bool show_title = 527232970; + // Text color for text tiles + string text_tile_text_color = 340996336; + // Background color for tiles + string tile_background_color = 467856637; + // Text color for tiles + string tile_text_color = 341304495; + // Color for titles + string title_color = 300393668; + // Warning button color + string warn_button_color = 359643012; + // The text alignment of tile titles (New Dashboards) + string tile_title_alignment = 490534298; + // Toggles the tile shadow (New Dashboards) + bool tile_shadow = 333903371; +} + + +message Timezone { + // Timezone + string value = 96868081; + // Description of timezone + string label = 102846452; + // Timezone group (e.g Common, Other, etc.) + string group = 107030207; +} + + +message UpdateFolder { + // Unique Name + string name = 3116757; + // Id of Parent. If the parent id is null, this is a root-level entry + string parent_id = 206134256; +} + + +message User { + // Operations the current user is able to perform on this object + map can = 108816; + // URL for the avatar image (may be generic) + string avatar_url = 119260393; + // URL for the avatar image (may be generic), does not specify size + string avatar_url_without_sizing = 490399578; + // API 3 credentials + repeated CredentialsApi3 credentials_api3 = 316399713; + CredentialsEmail credentials_email = 381333772; + // Embed credentials + repeated CredentialsEmbed credentials_embed = 518953269; + CredentialsGoogle credentials_google = 525839591; + CredentialsLDAP credentials_ldap = 443424330; + CredentialsLookerOpenid credentials_looker_openid = 335002066; + CredentialsOIDC credentials_oidc = 320440335; + CredentialsSaml credentials_saml = 493079932; + CredentialsTotp credentials_totp = 295786342; + // Full name for display (available only if both first_name and last_name are set) + string display_name = 322947240; + // EMail address + string email = 102965020; + // (DEPRECATED) (Embed only) ID of user's group space based on the external_group_id optionally specified during embed user login + string embed_group_space_id = 107104090; + // First name + string first_name = 277109009; + // Array of ids of the groups for this user + repeated string group_ids = 496513282; + // ID string for user's home folder + string home_folder_id = 115041580; + // Unique Id + string id = 3205; + // Account has been disabled + bool is_disabled = 332107427; + // Last name + string last_name = 510613627; + // User's preferred locale. User locale takes precedence over Looker's system-wide default locale. Locale determines language of display strings and date and numeric formatting in API responses. Locale string must be a 2 letter language code or a combination of language code and region code: 'en' or 'en-US', for example. + string locale = 325173590; + // Array of strings representing the Looker versions that this user has used (this only goes back as far as '3.54.0') + repeated string looker_versions = 484429195; + // User's dev workspace has been checked for presence of applicable production projects + bool models_dir_validated = 303600374; + // ID of user's personal folder + string personal_folder_id = 368241723; + // User is identified as an employee of Looker + bool presumed_looker_employee = 365833538; + // Array of ids of the roles for this user + repeated string role_ids = 212692081; + // Active sessions + repeated Session sessions = 339613953; + // Per user dictionary of undocumented state information owned by the Looker UI. + map ui_state = 401473497; + // User is identified as an employee of Looker who has been verified via Looker corporate authentication + bool verified_looker_employee = 397997346; + // User's roles are managed by an external directory like SAML or LDAP and can not be changed directly. + bool roles_externally_managed = 298138129; + // User can be directly assigned a role. + bool allow_direct_roles = 276166815; + // User can be a direct member of a normal Looker group. + bool allow_normal_group_membership = 215101880; + // User can inherit roles from a normal Looker group. + bool allow_roles_from_normal_groups = 301477131; + // (Embed only) ID of user's group folder based on the external_group_id optionally specified during embed user login + string embed_group_folder_id = 297056886; + // Link to get this item + string url = 107439; +} + + +message UserAttribute { + // Operations the current user is able to perform on this object + map can = 108816; + // Unique Id + string id = 3205; + // Name of user attribute + string name = 3116757; + // Human-friendly label for user attribute + string label = 102846452; + // Type of user attribute ("string", "number", "datetime", "yesno", "zipcode") + string type = 3120390; + // Default value for when no value is set on the user + string default_value = 429771091; + // Attribute is a system default + bool is_system = 529992220; + // Attribute is permanent and cannot be deleted + bool is_permanent = 468005635; + // If true, users will not be able to view values of this attribute + bool value_is_hidden = 527294216; + // Non-admin users can see the values of their attributes and use them in filters + bool user_can_view = 177700632; + // Users can change the value of this attribute for themselves + bool user_can_edit = 329711878; + // Destinations to which a hidden attribute may be sent. Once set, cannot be edited. + string hidden_value_domain_whitelist = 342844307; +} + +// An array of user attribute types that are allowed to be used in filters on this field. Valid values are: "advanced_filter_string", "advanced_filter_number", "advanced_filter_datetime", "string", "number", "datetime", "relative_url", "yesno", "zipcode". (Enum defined in LookmlModelExploreField) +enum UserAttributeFilterTypes { + _USER_ATTRIBUTE_FILTER_TYPES_UNSET = 0; + USER_ATTRIBUTE_FILTER_TYPES_ADVANCED_FILTER_STRING = 299892907; + USER_ATTRIBUTE_FILTER_TYPES_ADVANCED_FILTER_NUMBER = 527002578; + USER_ATTRIBUTE_FILTER_TYPES_ADVANCED_FILTER_DATETIME = 80041541; + USER_ATTRIBUTE_FILTER_TYPES_STRING = 493784209; + USER_ATTRIBUTE_FILTER_TYPES_NUMBER = 416901663; + USER_ATTRIBUTE_FILTER_TYPES_DATETIME = 433027416; + USER_ATTRIBUTE_FILTER_TYPES_RELATIVE_URL = 520555289; + USER_ATTRIBUTE_FILTER_TYPES_YESNO = 268804515; + USER_ATTRIBUTE_FILTER_TYPES_ZIPCODE = 469934548; +} + +// WARNING: no writeable properties found for POST, PUT, or PATCH +message UserAttributeGroupValue { + // Operations the current user is able to perform on this object + map can = 108816; + // Unique Id of this group-attribute relation + string id = 3205; + // Id of group + string group_id = 287220091; + // Id of user attribute + string user_attribute_id = 18993938; + // If true, the "value" field will be null, because the attribute settings block access to this value + bool value_is_hidden = 527294216; + // Precedence for resolving value for user + int64 rank = 3296468; + // Value of user attribute for group + string value = 96868081; +} + + +message UserAttributeWithValue { + // Operations the current user is able to perform on this object + map can = 108816; + // Name of user attribute + string name = 3116757; + // Human-friendly label for user attribute + string label = 102846452; + // Precedence for setting value on user (lowest wins) + int64 rank = 3296468; + // Value of attribute for user + string value = 96868081; + // Id of User + string user_id = 413336787; + // Can the user set this value + bool user_can_edit = 329711878; + // If true, the "value" field will be null, because the attribute settings block access to this value + bool value_is_hidden = 527294216; + // Id of User Attribute + string user_attribute_id = 18993938; + // How user got this value for this attribute + string source = 327120574; + // If this user attribute is hidden, whitelist of destinations to which it may be sent. + string hidden_value_domain_whitelist = 342844307; +} + + +message UserEmailOnly { + // Email Address + string email = 102965020; +} + + +message UserLoginLockout { + // Operations the current user is able to perform on this object + map can = 108816; + // Hash of user's client id + string key = 119519; + // Authentication method for login failures + string auth_type = 278079987; + // IP address of most recent failed attempt + string ip = 3577; + // User ID + string user_id = 413336787; + // Remote ID of user if using LDAP + string remote_id = 314868582; + // User's name + string full_name = 510674737; + // Email address associated with the user's account + string email = 102965020; + // Number of failures that triggered the lockout + int64 fail_count = 531780884; + // Time when lockout was triggered + google.protobuf.Timestamp lockout_at = 419861560; +} + + +message UserPublic { + // Operations the current user is able to perform on this object + map can = 108816; + // Unique Id + string id = 3205; + // First Name + string first_name = 277109009; + // Last Name + string last_name = 510613627; + // Full name for display (available only if both first_name and last_name are set) + string display_name = 322947240; + // URL for the avatar image (may be generic) + string avatar_url = 119260393; + // Link to get this item + string url = 107439; +} + + +message ValidationError { + // Error details + string message = 452371230; + // Error detail array + repeated ValidationErrorDetail errors = 446956773; + // Documentation link + string documentation_url = 402392181; +} + + +message ValidationErrorDetail { + // Field with error + string field = 95669946; + // Error code + string code = 3108531; + // Error info message + string message = 452371230; + // Documentation link + string documentation_url = 402392181; +} + +// The name of the starting day of the week. Valid values are: "monday", "tuesday", "wednesday", "thursday", "friday", "saturday", "sunday". (Enum defined in LookmlModelExploreField) +enum WeekStartDay { + _WEEK_START_DAY_UNSET = 0; + WEEK_START_DAY_MONDAY = 292419261; + WEEK_START_DAY_TUESDAY = 391195000; + WEEK_START_DAY_WEDNESDAY = 367089138; + WEEK_START_DAY_THURSDAY = 326497019; + WEEK_START_DAY_FRIDAY = 318966730; + WEEK_START_DAY_SATURDAY = 456678791; + WEEK_START_DAY_SUNDAY = 336631086; +} + + +message WelcomeEmailTest { + // The content that would be sent in the body of a custom welcome email + string content = 294427041; + // The subject that would be sent for the custom welcome email + string subject = 373234517; + // The header that would be sent in the body of a custom welcome email + string header = 467446454; +} + + +message WhitelabelConfiguration { + // Unique Id + string id = 3205; + // Customer logo image. Expected base64 encoded data (write-only) + string logo_file = 302348988; + // Logo image url (read-only) + string logo_url = 390957090; + // Custom favicon image. Expected base64 encoded data (write-only) + string favicon_file = 348157260; + // Favicon image url (read-only) + string favicon_url = 432048593; + // Default page title + string default_title = 491557590; + // Boolean to toggle showing help menus + bool show_help_menu = 319857861; + // Boolean to toggle showing docs + bool show_docs = 430915951; + // Boolean to toggle showing email subscription options. + bool show_email_sub_options = 481532061; + // Boolean to toggle mentions of Looker in emails + bool allow_looker_mentions = 373852627; + // Boolean to toggle links to Looker in emails + bool allow_looker_links = 375199970; + // Allow subject line and email heading customization in customized emails” + bool custom_welcome_email_advanced = 119812016; + // Remove the word Looker from appearing in the account setup page + bool setup_mentions = 338894507; + // Remove Looker logo from Alerts + bool alerts_logo = 409573540; + // Remove Looker links from Alerts + bool alerts_links = 304998460; + // Remove Looker mentions in home folder page when you don’t have any items saved + bool folders_mentions = 496168113; +} + + +message Workspace { + // Operations the current user is able to perform on this object + map can = 108816; + // The unique id of this user workspace. Predefined workspace ids include "production" and "dev" + string id = 3205; + // The local state of each project in the workspace + repeated Project projects = 340441809; +} + +// Dynamic writeable type for Alert removes: + // followed, followable, id, investigative_content_title, owner_display_name +message WriteAlert { + // Filters coming from the dashboard that are applied. Example `[{ "filter_title": "Name", "field_name": "distribution_centers.name", "filter_value": "Los Angeles CA" }]` + repeated AlertAppliedDashboardFilter applied_dashboard_filters = 337449948; + // This property informs the check what kind of comparison we are performing. Only certain condition types are valid for time series alerts. For details, refer to [Setting Alert Conditions](https://docs.looker.com/sharing-and-publishing/creating-alerts#setting_alert_conditions) Valid values are: "EQUAL_TO", "GREATER_THAN", "GREATER_THAN_OR_EQUAL_TO", "LESS_THAN", "LESS_THAN_OR_EQUAL_TO", "INCREASES_BY", "DECREASES_BY", "CHANGES_BY". + ComparisonType comparison_type = 300174680; + // Vixie-Style crontab specification when to run. At minumum, it has to be longer than 15 minute intervals + string cron = 3387314; + // An optional, user-defined title for the alert + string custom_title = 413477701; + // ID of the dashboard element associated with the alert. Refer to [dashboard_element()](#!/Dashboard/DashboardElement) + string dashboard_element_id = 426786746; + // An optional description for the alert. This supplements the title + string description = 375083905; + // Array of destinations to send alerts to. Must be the same type of destination. Example `[{ "destination_type": "EMAIL", "email_address": "test@test.com" }]` + repeated AlertDestination destinations = 166962363; + AlertField field = 95669946; + // Whether or not the alert is disabled + bool is_disabled = 332107427; + // Reason for disabling alert + string disabled_reason = 367441865; + // Whether or not the alert is public + bool is_public = 254583906; + // The type of the investigative content Valid values are: "dashboard". + InvestigativeContentType investigative_content_type = 455691579; + // The ID of the investigative content. For dashboards, this will be the dashboard ID + string investigative_content_id = 329023657; + // ID of the LookML dashboard associated with the alert + string lookml_dashboard_id = 351003409; + // ID of the LookML dashboard element associated with the alert + string lookml_link_id = 527451937; + // User id of alert owner + string owner_id = 285850503; + // Value of the alert threshold + double threshold = 425888618; + AlertConditionState time_series_condition_state = 343695767; +} + +// Dynamic writeable type for ApiSession removes: + // can, sudo_user_id +message WriteApiSession { + // The id of active workspace for this session + string workspace_id = 456431629; +} + +// Dynamic writeable type for BackupConfiguration removes: + // can, url +message WriteBackupConfiguration { + // Type of backup: looker-s3 or custom-s3 + string type = 3120390; + // Name of bucket for custom-s3 backups + string custom_s3_bucket = 244474693; + // Name of region where the bucket is located + string custom_s3_bucket_region = 511205324; + // (Write-Only) AWS S3 key used for custom-s3 backups + string custom_s3_key = 515118471; + // (Write-Only) AWS S3 secret used for custom-s3 backups + string custom_s3_secret = 387048480; +} + +// Dynamic writeable type for Board removes: + // can, content_metadata_id, created_at, board_sections, id, updated_at, user_id, primary_homepage +message WriteBoard { + // Date of board deletion + google.protobuf.Timestamp deleted_at = 464526556; + // Description of the board + string description = 375083905; + // ids of the board sections in the order they should be displayed + repeated string section_order = 511535011; + // Title of the board + string title = 96607896; +} + +// Dynamic writeable type for BoardItem removes: + // can, content_created_by, content_favorite_id, content_metadata_id, content_updated_at, description, favorite_count, id, image_url, location, title, url, view_count, custom_image_url +message WriteBoardItem { + // Custom description entered by the user, if present + string custom_description = 517484699; + // Custom title entered by the user, if present + string custom_title = 413477701; + // Custom url entered by the user, if present + string custom_url = 249459585; + // Dashboard to base this item on + string dashboard_id = 311679049; + // Associated Board Section + string board_section_id = 473711058; + // Look to base this item on + string look_id = 413287022; + // LookML Dashboard to base this item on + string lookml_dashboard_id = 351003409; + // An arbitrary integer representing the sort order within the section + int64 order = 108390030; + // Whether the custom description should be used instead of the content description, if the item is associated with content + bool use_custom_description = 532957353; + // Whether the custom title should be used instead of the content title, if the item is associated with content + bool use_custom_title = 533570657; + // Whether the custom url should be used instead of the content url, if the item is associated with content + bool use_custom_url = 441644771; + // (Write-Only) base64 encoded image data + string custom_image_data_base64 = 427605623; + // Whether the custom image should be used instead of the content image, if the item is associated with content + bool use_custom_image = 398202105; +} + +// Dynamic writeable type for BoardSection removes: + // can, created_at, board_items, id, visible_item_order, updated_at +message WriteBoardSection { + // Time at which this section was deleted. + google.protobuf.Timestamp deleted_at = 464526556; + // Description of the content found in this section. + string description = 375083905; + // Id reference to parent board + string board_id = 298405268; + // ids of the board items in the order they should be displayed + repeated string item_order = 377743503; + // Name of row + string title = 96607896; +} + +// Dynamic writeable type for ColorCollection removes: + // id +message WriteColorCollection { + // Label of color collection + string label = 102846452; + // Array of categorical palette definitions + repeated DiscretePalette categoricalPalettes = 298797876; + // Array of discrete palette definitions + repeated ContinuousPalette sequentialPalettes = 421922795; + // Array of diverging palette definitions + repeated ContinuousPalette divergingPalettes = 402046609; +} + +// Dynamic writeable type for ContentFavorite removes: + // id, look_id, dashboard_id, board_id +message WriteContentFavorite { + // User Id which owns this ContentFavorite + string user_id = 413336787; + // Content Metadata Id associated with this ContentFavorite + string content_metadata_id = 4111259; + // Dynamic writeable type for LookBasic removes: + // can, content_metadata_id, id, title + WriteLookBasic look = 3297857; + // Dynamic writeable type for DashboardBase removes: + // can, content_favorite_id, content_metadata_id, description, hidden, id, model, query_timezone, readonly, refresh_interval, refresh_interval_to_i, title, user_id, slug, preferred_viewer + WriteDashboardBase dashboard = 79840748; +} + +// Dynamic writeable type for ContentMeta removes: + // can, id, name, parent_id, dashboard_id, look_id, folder_id, content_type, inheriting_id, slug +message WriteContentMeta { + // Whether content inherits its access levels from parent + bool inherits = 386348026; +} + +// Dynamic writeable type for CreateDashboardFilter removes: + // id, field +message WriteCreateDashboardFilter { + // Id of Dashboard + string dashboard_id = 311679049; + // Name of filter + string name = 3116757; + // Title of filter + string title = 96607896; + // Type of filter: one of date, number, string, or field + string type = 3120390; + // Default value of filter + string default_value = 429771091; + // Model of filter (required if type = field) + string model = 102848809; + // Explore of filter (required if type = field) + string explore = 370461451; + // Dimension of filter (required if type = field) + string dimension = 511293107; + // Display order of this filter relative to other filters + int64 row = 117914; + // Array of listeners for faceted filters + repeated string listens_to_filters = 493150066; + // Whether the filter allows multiple filter values (deprecated in the latest version of dashboards) + bool allow_multiple_values = 331288581; + // Whether the filter requires a value to run the dashboard + bool required = 497611616; + // The visual configuration for this filter. Used to set up how the UI for this filter should appear. + map ui_config = 413099149; +} + +// Dynamic writeable type for CreateQueryTask removes: + // can +message WriteCreateQueryTask { + // Id of query to run + string query_id = 279007282; + // Desired async query result format. Valid values are: "inline_json", "json", "json_detail", "json_fe", "csv", "html", "md", "txt", "xlsx", "gsxml". + ResultFormat result_format = 296518982; + // Source of query task + string source = 327120574; + // Create the task but defer execution + bool deferred = 501523840; + // Id of look associated with query. + string look_id = 413287022; + // Id of dashboard associated with query. + string dashboard_id = 311679049; +} + +// Dynamic writeable type for CredentialsEmail removes: + // can, created_at, is_disabled, logged_in_at, password_reset_url, type, url, user_url +message WriteCredentialsEmail { + // EMail address used for user login + string email = 102965020; + // Force the user to change their password upon their next login + bool forced_password_reset_at_next_login = 458418973; +} + +// Dynamic writeable type for Dashboard removes: + // can, content_favorite_id, content_metadata_id, id, model, readonly, refresh_interval_to_i, user_id, created_at, dashboard_elements, dashboard_filters, dashboard_layouts, deleted_at, deleter_id, edit_uri, favorite_count, last_accessed_at, last_viewed_at, updated_at, last_updater_id, last_updater_name, user_name, view_count, url +message WriteDashboard { + // Description + string description = 375083905; + // Is Hidden + bool hidden = 524703253; + // Timezone in which the Dashboard will run by default. + string query_timezone = 391786756; + // Refresh Interval, as a time duration phrase like "2 hours 30 minutes". A number with no time units will be interpreted as whole seconds. + string refresh_interval = 382963810; + // Dynamic writeable type for FolderBase removes: + // id, content_metadata_id, created_at, creator_id, child_count, external_id, is_embed, is_embed_shared_root, is_embed_users_root, is_personal, is_personal_descendant, is_shared_root, is_users_root, can + WriteFolderBase folder = 467441015; + // Dashboard Title + string title = 96607896; + // Content Metadata Slug + string slug = 3184373; + // The preferred route for viewing this dashboard (ie: dashboards or dashboards-next) + string preferred_viewer = 351399076; + // Enables alerts to keep in sync with dashboard filter changes + bool alert_sync_with_dashboard_filter_enabled = 440126712; + // Background color + string background_color = 351188052; + // Enables crossfiltering in dashboards - only available in dashboards-next (beta) + bool crossfilter_enabled = 450959918; + // Whether or not a dashboard is 'soft' deleted. + bool deleted = 389460870; + // Sets the default state of the filters bar to collapsed or open + bool filters_bar_collapsed = 324566887; + // configuration option that governs how dashboard loading will happen. + string load_configuration = 446683328; + // Links this dashboard to a particular LookML dashboard such that calling a **sync** operation on that LookML dashboard will update this dashboard to match. + string lookml_link_id = 527451937; + // Show filters bar. **Security Note:** This property only affects the *cosmetic* appearance of the dashboard, not a user's ability to access data. Hiding the filters bar does **NOT** prevent users from changing filters by other means. For information on how to set up secure data access control policies, see [Control User Access to Data](https://looker.com/docs/r/api/control-access) + bool show_filters_bar = 70091731; + // Show title + bool show_title = 527232970; + // Id of folder + string folder_id = 271739380; + // Color of text on text tiles + string text_tile_text_color = 340996336; + // Tile background color + string tile_background_color = 467856637; + // Tile text color + string tile_text_color = 341304495; + // Title color + string title_color = 300393668; + DashboardAppearance appearance = 381592142; +} + +// Dynamic writeable type for DashboardBase removes: + // can, content_favorite_id, content_metadata_id, description, hidden, id, model, query_timezone, readonly, refresh_interval, refresh_interval_to_i, title, user_id, slug, preferred_viewer +message WriteDashboardBase { + // Dynamic writeable type for FolderBase removes: + // id, content_metadata_id, created_at, creator_id, child_count, external_id, is_embed, is_embed_shared_root, is_embed_users_root, is_personal, is_personal_descendant, is_shared_root, is_users_root, can + WriteFolderBase folder = 467441015; +} + +// Dynamic writeable type for DashboardElement removes: + // can, body_text_as_html, edit_uri, id, lookml_link_id, note_text_as_html, refresh_interval_to_i, alert_count, title_text_as_html, subtitle_text_as_html +message WriteDashboardElement { + // Text tile body text + string body_text = 491616469; + // Id of Dashboard + string dashboard_id = 311679049; + // Dynamic writeable type for LookWithQuery removes: + // can, content_metadata_id, id, content_favorite_id, created_at, deleted_at, deleter_id, embed_url, excel_file_url, favorite_count, google_spreadsheet_formula, image_embed_url, last_accessed_at, last_updater_id, last_viewed_at, model, public_slug, public_url, short_url, updated_at, view_count, url + WriteLookWithQuery look = 3297857; + // Id Of Look + string look_id = 413287022; + // ID of merge result + string merge_result_id = 3223242; + // Note Display + string note_display = 120563243; + // Note State + string note_state = 342745425; + // Note Text + string note_text = 491761577; + // Dynamic writeable type for Query removes: + // can, id, slug, share_url, expanded_share_url, url, has_table_calculations + WriteQuery query = 115243016; + // Id Of Query + string query_id = 279007282; + // Refresh Interval + string refresh_interval = 382963810; + // Dynamic writeable type for ResultMakerWithIdVisConfigAndDynamicFields removes: + // id, dynamic_fields, filterables, sorts, merge_result_id, total, query_id, sql_query_id, vis_config + WriteResultMakerWithIdVisConfigAndDynamicFields result_maker = 476157982; + // ID of the ResultMakerLookup entry. + string result_maker_id = 388730214; + // Text tile subtitle text + string subtitle_text = 507090106; + // Title of dashboard element + string title = 96607896; + // Whether title is hidden + bool title_hidden = 308186743; + // Text tile title + string title_text = 424563514; + // Type + string type = 3120390; + // JSON with all the properties required for rich editor and buttons elements + string rich_content_json = 309924280; +} + +// Dynamic writeable type for DashboardFilter removes: + // can, id, dashboard_id, field +message WriteDashboardFilter { + // Name of filter + string name = 3116757; + // Title of filter + string title = 96607896; + // Type of filter: one of date, number, string, or field + string type = 3120390; + // Default value of filter + string default_value = 429771091; + // Model of filter (required if type = field) + string model = 102848809; + // Explore of filter (required if type = field) + string explore = 370461451; + // Dimension of filter (required if type = field) + string dimension = 511293107; + // Display order of this filter relative to other filters + int64 row = 117914; + // Array of listeners for faceted filters + repeated string listens_to_filters = 493150066; + // Whether the filter allows multiple filter values (deprecated in the latest version of dashboards) + bool allow_multiple_values = 331288581; + // Whether the filter requires a value to run the dashboard + bool required = 497611616; + // The visual configuration for this filter. Used to set up how the UI for this filter should appear. + map ui_config = 413099149; +} + +// Dynamic writeable type for DashboardLayout removes: + // can, id, deleted, dashboard_title, dashboard_layout_components +message WriteDashboardLayout { + // Id of Dashboard + string dashboard_id = 311679049; + // Type + string type = 3120390; + // Is Active + bool active = 322801217; + // Column Width + int64 column_width = 530224687; + // Width + int64 width = 99601414; +} + +// Dynamic writeable type for DashboardLayoutComponent removes: + // can, id, deleted, element_title, element_title_hidden, vis_type +message WriteDashboardLayoutComponent { + // Id of Dashboard Layout + string dashboard_layout_id = 446158738; + // Id Of Dashboard Element + string dashboard_element_id = 426786746; + // Row + int64 row = 117914; + // Column + int64 column = 520752011; + // Width + int64 width = 99601414; + // Height + int64 height = 437383491; +} + +// Dynamic writeable type for DashboardLookml removes: + // dashboard_id +message WriteDashboardLookml { + // (Write-Only) Id of the folder + string folder_id = 271739380; + // lookml of UDD + string lookml = 274764335; +} + +// Dynamic writeable type for Datagroup removes: + // can, created_at, id, model_name, name, trigger_check_at, trigger_error, trigger_value +message WriteDatagroup { + // UNIX timestamp before which cache entries are considered stale. Cannot be in the future. + int64 stale_before = 485547741; + // UNIX timestamp at which this entry became triggered. Cannot be in the future. + int64 triggered_at = 311600238; +} + +// Dynamic writeable type for DBConnection removes: + // can, dialect, snippets, pdts_enabled, uses_oauth, created_at, user_id, example, last_regen_at, last_reap_at, managed +message WriteDBConnection { + // Name of the connection. Also used as the unique identifier + string name = 3116757; + // Host name/address of server + string host = 3569816; + // Port number on server + string port = 3568863; + // Username for server authentication + string username = 382974997; + // (Write-Only) Password for server authentication + string password = 425146747; + // (Write-Only) Base64 encoded Certificate body for server authentication (when appropriate for dialect). + string certificate = 402617131; + // (Write-Only) Certificate keyfile type - .json or .p12 + string file_type = 278104344; + // Database name + string database = 446107481; + // Time zone of database + string db_timezone = 530059610; + // Timezone to use in queries + string query_timezone = 391786756; + // Scheme name + string schema = 353540960; + // Maximum number of concurrent connection to use + int64 max_connections = 481127633; + // Maximum size of query in GBs (BigQuery only, can be a user_attribute name) + string max_billing_gigabytes = 367356460; + // Use SSL/TLS when connecting to server + bool ssl = 107468; + // Verify the SSL + bool verify_ssl = 456329219; + // Name of temporary database (if used) + string tmp_db_name = 527957692; + // Additional params to add to JDBC connection string + string jdbc_additional_params = 453083961; + // Connection Pool Timeout, in seconds + int64 pool_timeout = 350051759; + // (Read/Write) SQL Dialect name + string dialect_name = 414841156; + // (Limited access feature) Are per user db credentials enabled. Enabling will remove previously set username and password + bool user_db_credentials = 401568052; + // Fields whose values map to user attribute names + repeated string user_attribute_fields = 424067660; + // Cron string specifying when maintenance such as PDT trigger checks and drops should be performed + string maintenance_cron = 291048785; + // Precache tables in the SQL Runner + bool sql_runner_precache_tables = 306935836; + // Fetch Information Schema For SQL Writing + bool sql_writing_with_info_schema = 462710732; + // SQL statements (semicolon separated) to issue after connecting to the database. Requires `custom_after_connect_statements` license feature + string after_connect_statements = 84658980; + // Dynamic writeable type for DBConnectionOverride removes: + // has_password + WriteDBConnectionOverride pdt_context_override = 442814086; + // The Id of the ssh tunnel this connection uses + string tunnel_id = 443214254; + // Maximum number of threads to use to build PDTs in parallel + int64 pdt_concurrency = 312616502; + // When disable_context_comment is true comment will not be added to SQL + bool disable_context_comment = 315519886; + // An External OAuth Application to use for authenticating to the database + string oauth_application_id = 254423058; + // When true, error PDTs will be retried every regenerator cycle + bool always_retry_failed_builds = 297083520; + // When true, query cost estimate will be displayed in explore. + bool cost_estimate_enabled = 468369479; + // PDT builds on this connection can be kicked off and cancelled via API. + bool pdt_api_control_enabled = 521873634; +} + +// Dynamic writeable type for DBConnectionOverride removes: + // has_password +message WriteDBConnectionOverride { + // Context in which to override (`pdt` is the only allowed value) + string context = 445708328; + // Host name/address of server + string host = 3569816; + // Port number on server + string port = 3568863; + // Username for server authentication + string username = 382974997; + // (Write-Only) Password for server authentication + string password = 425146747; + // (Write-Only) Base64 encoded Certificate body for server authentication (when appropriate for dialect). + string certificate = 402617131; + // (Write-Only) Certificate keyfile type - .json or .p12 + string file_type = 278104344; + // Database name + string database = 446107481; + // Scheme name + string schema = 353540960; + // Additional params to add to JDBC connection string + string jdbc_additional_params = 453083961; + // SQL statements (semicolon separated) to issue after connecting to the database. Requires `custom_after_connect_statements` license feature + string after_connect_statements = 84658980; +} + +// Dynamic writeable type for EmbedSecret removes: + // created_at, id, secret, user_id +message WriteEmbedSecret { + // Signing algorithm to use with this secret. Either `hmac/sha-256`(default) or `hmac/sha-1` + string algorithm = 110603631; + // Is this secret currently enabled + bool enabled = 387588912; +} + +// Dynamic writeable type for ExternalOauthApplication removes: + // can, id, created_at +message WriteExternalOauthApplication { + // The name of this application. For Snowflake connections, this should be the name of the host database. + string name = 3116757; + // The OAuth Client ID for this application + string client_id = 206142577; + // (Write-Only) The OAuth Client Secret for this application + string client_secret = 509513545; + // The database dialect for this application. + string dialect_name = 414841156; +} + +// Dynamic writeable type for FolderBase removes: + // id, content_metadata_id, created_at, creator_id, child_count, external_id, is_embed, is_embed_shared_root, is_embed_users_root, is_personal, is_personal_descendant, is_shared_root, is_users_root, can +message WriteFolderBase { + // Unique Name + string name = 3116757; + // Id of Parent. If the parent id is null, this is a root-level entry + string parent_id = 206134256; +} + +// Dynamic writeable type for GitBranch removes: + // can, remote, remote_name, error, message, owner_name, readonly, personal, is_local, is_remote, is_production, ahead_count, behind_count, commit_at, remote_ref +message WriteGitBranch { + // The short name on the local. Updating `name` results in `git checkout ` + string name = 3116757; + // The resolved ref of this branch. Updating `ref` results in `git reset --hard ``. + string ref = 101267; +} + +// Dynamic writeable type for Group removes: + // can, contains_current_user, external_group_id, externally_managed, id, include_by_default, user_count +message WriteGroup { + // Group can be used in content access controls + bool can_add_to_content_metadata = 454846998; + // Name of group + string name = 3116757; +} + +// Dynamic writeable type for Integration removes: + // can, id, integration_hub_id, label, description, supported_formats, supported_action_types, supported_formattings, supported_visualization_formattings, supported_download_settings, icon_url, uses_oauth, required_fields, delegate_oauth +message WriteIntegration { + // Whether the integration is available to users. + bool enabled = 387588912; + // Array of params for the integration. + repeated IntegrationParam params = 449474371; + // Whether the integration is available to users. + repeated string installed_delegate_oauth_targets = 359534361; +} + +// Dynamic writeable type for IntegrationHub removes: + // can, id, label, official, fetch_error_message, has_authorization_token, legal_agreement_signed, legal_agreement_required, legal_agreement_text +message WriteIntegrationHub { + // URL of the hub. + string url = 107439; + // (Write-Only) An authorization key that will be sent to the integration hub on every request. + string authorization_token = 290870323; +} + +// Dynamic writeable type for InternalHelpResources removes: + // can +message WriteInternalHelpResources { + // If true and internal help resources content is not blank then the link for internal help resources will be shown in the help menu and the content displayed within Looker + bool enabled = 387588912; +} + +// Dynamic writeable type for InternalHelpResourcesContent removes: + // can +message WriteInternalHelpResourcesContent { + // Text to display in the help menu item which will display the internal help resources + string organization_name = 71465175; + // Content to be displayed in the internal help resources page/modal + string markdown_content = 379297421; +} + +// Dynamic writeable type for LDAPConfig removes: + // can, default_new_user_groups, default_new_user_roles, groups, has_auth_password, modified_at, modified_by, user_attributes, url +message WriteLDAPConfig { + // Allow alternate email-based login via '/login/email' for admins and for specified users with the 'login_special_email' permission. This option is useful as a fallback during ldap setup, if ldap config problems occur later, or if you need to support some users who are not in your ldap directory. Looker email/password logins are always disabled for regular users when ldap is enabled. + bool alternate_email_login_allowed = 415381803; + // (Write-Only) Password for the LDAP account used to access the LDAP server + string auth_password = 450780407; + // Users will not be allowed to login at all unless a role for them is found in LDAP if set to true + bool auth_requires_role = 359416088; + // Distinguished name of LDAP account used to access the LDAP server + string auth_username = 358003939; + // LDAP server hostname + string connection_host = 526430378; + // LDAP host port + string connection_port = 325512703; + // Use Transport Layer Security + bool connection_tls = 364401289; + // Do not verify peer when using TLS + bool connection_tls_no_verify = 374286647; + // (Write-Only) Array of ids of groups that will be applied to new users the first time they login via LDAP + repeated string default_new_user_group_ids = 417958896; + // (Write-Only) Array of ids of roles that will be applied to new users the first time they login via LDAP + repeated string default_new_user_role_ids = 319694385; + // Enable/Disable LDAP authentication for the server + bool enabled = 387588912; + // Don't attempt to do LDAP search result paging (RFC 2696) even if the LDAP server claims to support it. + bool force_no_page = 272379335; + // Base dn for finding groups in LDAP searches + string groups_base_dn = 321940973; + // Identifier for a strategy for how Looker will search for groups in the LDAP server + string groups_finder_type = 21587272; + // LDAP Group attribute that signifies the members of the groups. Most commonly 'member' + string groups_member_attribute = 450619343; + // Optional comma-separated list of supported LDAP objectclass for groups when doing groups searches + string groups_objectclasses = 402805559; + // LDAP Group attribute that signifies the user in a group. Most commonly 'dn' + string groups_user_attribute = 394608153; + // (Read/Write) Array of mappings between LDAP Groups and arrays of Looker Role ids + repeated LDAPGroupWrite groups_with_role_ids = 475375615; + // Merge first-time ldap login to existing user account by email addresses. When a user logs in for the first time via ldap this option will connect this user into their existing account by finding the account with a matching email address. Otherwise a new user account will be created for the user. + bool merge_new_users_by_email = 469551208; + // Set user roles in Looker based on groups from LDAP + bool set_roles_from_groups = 390938262; + // (Write-Only) Test LDAP user password. For ldap tests only. + string test_ldap_password = 494187581; + // (Write-Only) Test LDAP user login id. For ldap tests only. + string test_ldap_user = 507373170; + // Name of user record attributes used to indicate email address field + string user_attribute_map_email = 398964231; + // Name of user record attributes used to indicate first name + string user_attribute_map_first_name = 400740186; + // Name of user record attributes used to indicate last name + string user_attribute_map_last_name = 521517530; + // Name of user record attributes used to indicate unique record id + string user_attribute_map_ldap_id = 466191369; + // (Read/Write) Array of mappings between LDAP User Attributes and arrays of Looker User Attribute ids + repeated LDAPUserAttributeWrite user_attributes_with_ids = 332652790; + // Distinguished name of LDAP node used as the base for user searches + string user_bind_base_dn = 289800939; + // (Optional) Custom RFC-2254 filter clause for use in finding user during login. Combined via 'and' with the other generated filter clauses. + string user_custom_filter = 129489070; + // Name(s) of user record attributes used for matching user login id (comma separated list) + string user_id_attribute_names = 236102283; + // (Optional) Name of user record objectclass used for finding user during login id + string user_objectclass = 358011678; + // Allow LDAP auth'd users to be members of non-reflected Looker groups. If 'false', user will be removed from non-reflected groups on login. + bool allow_normal_group_membership = 215101880; + // LDAP auth'd users will be able to inherit roles from non-reflected Looker groups. + bool allow_roles_from_normal_groups = 301477131; + // Allows roles to be directly assigned to LDAP auth'd users. + bool allow_direct_roles = 276166815; +} + +// Dynamic writeable type for LegacyFeature removes: + // can, id, name, description, enabled, disallowed_as_of_version, disable_on_upgrade_to_version, end_of_life_version, documentation_url, approximate_disable_date, approximate_end_of_life_date, has_disabled_on_upgrade +message WriteLegacyFeature { + // Whether this feature has been enabled by a user + bool enabled_locally = 276773134; +} + +// Dynamic writeable type for LookBasic removes: + // can, content_metadata_id, id, title +message WriteLookBasic { + // User Id + string user_id = 413336787; +} + +// Dynamic writeable type for LookmlModel removes: + // can, explores, has_content, label +message WriteLookmlModel { + // Array of names of connections this model is allowed to use + repeated string allowed_db_connection_names = 350456152; + // Name of the model. Also used as the unique identifier + string name = 3116757; + // Name of project containing the model + string project_name = 414886993; + // Is this model allowed to use all current and future connections + bool unlimited_db_connections = 413404803; +} + +// Dynamic writeable type for LookWithQuery removes: + // can, content_metadata_id, id, content_favorite_id, created_at, deleted_at, deleter_id, embed_url, excel_file_url, favorite_count, google_spreadsheet_formula, image_embed_url, last_accessed_at, last_updater_id, last_viewed_at, model, public_slug, public_url, short_url, updated_at, view_count, url +message WriteLookWithQuery { + // Look Title + string title = 96607896; + // User Id + string user_id = 413336787; + // Whether or not a look is 'soft' deleted. + bool deleted = 389460870; + // Description + string description = 375083905; + // auto-run query when Look viewed + bool is_run_on_load = 487054535; + // Is Public + bool public = 340099074; + // Query Id + string query_id = 279007282; + // Dynamic writeable type for FolderBase removes: + // id, content_metadata_id, created_at, creator_id, child_count, external_id, is_embed, is_embed_shared_root, is_embed_users_root, is_personal, is_personal_descendant, is_shared_root, is_users_root, can + WriteFolderBase folder = 467441015; + // Folder Id + string folder_id = 271739380; + // Dynamic writeable type for Query removes: + // can, id, slug, share_url, expanded_share_url, url, has_table_calculations + WriteQuery query = 115243016; +} + +// Dynamic writeable type for MergeQuery removes: + // can, id, result_maker_id +message WriteMergeQuery { + // Column Limit + string column_limit = 436244100; + // Dynamic Fields + string dynamic_fields = 405484086; + // Pivots + repeated string pivots = 446031464; + // Sorts + repeated string sorts = 109773781; + // Source Queries defining the results to be merged. + repeated MergeQuerySourceQuery source_queries = 369604511; + // Total + bool total = 102745028; + // Visualization Config + map vis_config = 78818913; +} + +// Dynamic writeable type for ModelSet removes: + // can, all_access, built_in, id, url +message WriteModelSet { + repeated string models = 449883061; + // Name of ModelSet + string name = 3116757; +} + +// Dynamic writeable type for OauthClientApp removes: + // can, client_guid, tokens_invalid_before, activated_users +message WriteOauthClientApp { + // The uri with which this application will receive an auth code by browser redirect. + string redirect_uri = 379793333; + // The application's display name + string display_name = 322947240; + // A description of the application that will be displayed to users + string description = 375083905; + // When enabled is true, OAuth2 and API requests will be accepted from this app. When false, all requests from this app will be refused. + bool enabled = 387588912; + // If set, only Looker users who are members of this group can use this web app with Looker. If group_id is not set, any Looker user may use this app to access this Looker instance + string group_id = 287220091; +} + +// Dynamic writeable type for OIDCConfig removes: + // can, default_new_user_groups, default_new_user_roles, groups, modified_at, modified_by, test_slug, user_attributes, url +message WriteOIDCConfig { + // Allow alternate email-based login via '/login/email' for admins and for specified users with the 'login_special_email' permission. This option is useful as a fallback during ldap setup, if ldap config problems occur later, or if you need to support some users who are not in your ldap directory. Looker email/password logins are always disabled for regular users when ldap is enabled. + bool alternate_email_login_allowed = 415381803; + // OpenID Provider Audience + string audience = 422223006; + // Users will not be allowed to login at all unless a role for them is found in OIDC if set to true + bool auth_requires_role = 359416088; + // OpenID Provider Authorization Url + string authorization_endpoint = 3578117; + // (Write-Only) Array of ids of groups that will be applied to new users the first time they login via OIDC + repeated string default_new_user_group_ids = 417958896; + // (Write-Only) Array of ids of roles that will be applied to new users the first time they login via OIDC + repeated string default_new_user_role_ids = 319694385; + // Enable/Disable OIDC authentication for the server + bool enabled = 387588912; + // Name of user record attributes used to indicate groups. Used when 'groups_finder_type' is set to 'grouped_attribute_values' + string groups_attribute = 389492843; + // (Read/Write) Array of mappings between OIDC Groups and arrays of Looker Role ids + repeated OIDCGroupWrite groups_with_role_ids = 475375615; + // Relying Party Identifier (provided by OpenID Provider) + string identifier = 333214290; + // OpenID Provider Issuer + string issuer = 467184364; + // Merge first-time oidc login to existing user account by email addresses. When a user logs in for the first time via oidc this option will connect this user into their existing account by finding the account with a matching email address by testing the given types of credentials for existing users. Otherwise a new user account will be created for the user. This list (if provided) must be a comma separated list of string like 'email,ldap,google' + string new_user_migration_types = 331422246; + // Array of scopes to request. + repeated string scopes = 452946431; + // (Write-Only) Relying Party Secret (provided by OpenID Provider) + string secret = 438607800; + // Set user roles in Looker based on groups from OIDC + bool set_roles_from_groups = 390938262; + // OpenID Provider Token Url + string token_endpoint = 534249150; + // Name of user record attributes used to indicate email address field + string user_attribute_map_email = 398964231; + // Name of user record attributes used to indicate first name + string user_attribute_map_first_name = 400740186; + // Name of user record attributes used to indicate last name + string user_attribute_map_last_name = 521517530; + // (Read/Write) Array of mappings between OIDC User Attributes and arrays of Looker User Attribute ids + repeated OIDCUserAttributeWrite user_attributes_with_ids = 332652790; + // OpenID Provider User Information Url + string userinfo_endpoint = 274918578; + // Allow OIDC auth'd users to be members of non-reflected Looker groups. If 'false', user will be removed from non-reflected groups on login. + bool allow_normal_group_membership = 215101880; + // OIDC auth'd users will inherit roles from non-reflected Looker groups. + bool allow_roles_from_normal_groups = 301477131; + // Allows roles to be directly assigned to OIDC auth'd users. + bool allow_direct_roles = 276166815; +} + +// Dynamic writeable type for PasswordConfig removes: + // can +message WritePasswordConfig { + // Minimum number of characters required for a new password. Must be between 7 and 100 + int64 min_length = 378159156; + // Require at least one numeric character + bool require_numeric = 438215331; + // Require at least one uppercase and one lowercase letter + bool require_upperlower = 330053382; + // Require at least one special character + bool require_special = 301715528; +} + +// Dynamic writeable type for PermissionSet removes: + // can, all_access, built_in, id, url +message WritePermissionSet { + // Name of PermissionSet + string name = 3116757; + repeated string permissions = 322818529; +} + +// Dynamic writeable type for PrivatelabelConfiguration removes: + // logo_url, favicon_url +message WritePrivatelabelConfiguration { + // Customer logo image. Expected base64 encoded data (write-only) + string logo_file = 302348988; + // Custom favicon image. Expected base64 encoded data (write-only) + string favicon_file = 348157260; + // Default page title + string default_title = 491557590; + // Boolean to toggle showing help menus + bool show_help_menu = 319857861; + // Boolean to toggle showing docs + bool show_docs = 430915951; + // Boolean to toggle showing email subscription options. + bool show_email_sub_options = 481532061; + // Boolean to toggle mentions of Looker in emails + bool allow_looker_mentions = 373852627; + // Boolean to toggle links to Looker in emails + bool allow_looker_links = 375199970; + // Allow subject line and email heading customization in customized emails” + bool custom_welcome_email_advanced = 119812016; + // Remove the word Looker from appearing in the account setup page + bool setup_mentions = 338894507; + // Remove Looker logo from Alerts + bool alerts_logo = 409573540; + // Remove Looker links from Alerts + bool alerts_links = 304998460; + // Remove Looker mentions in home folder page when you don’t have any items saved + bool folders_mentions = 496168113; +} + +// Dynamic writeable type for Project removes: + // can, id, uses_git, is_example +message WriteProject { + // Project display name + string name = 3116757; + // Git remote repository url + string git_remote_url = 409688477; + // Git username for HTTPS authentication. (For production only, if using user attributes.) + string git_username = 404090688; + // (Write-Only) Git password for HTTPS authentication. (For production only, if using user attributes.) + string git_password = 269826806; + // Git production branch name. Defaults to master. Supported only in Looker 21.0 and higher. + string git_production_branch_name = 384108369; + // If true, the project uses a git cookie for authentication. + bool use_git_cookie_auth = 517923168; + // User attribute name for username in per-user HTTPS authentication. + string git_username_user_attribute = 312364062; + // User attribute name for password in per-user HTTPS authentication. + string git_password_user_attribute = 517735463; + // Name of the git service provider + string git_service_name = 365333079; + // Port that HTTP(S) application server is running on (for PRs, file browsing, etc.) + int64 git_application_server_http_port = 529772882; + // Scheme that is running on application server (for PRs, file browsing, etc.) + string git_application_server_http_scheme = 330358350; + // (Write-Only) Optional secret token with which to authenticate requests to the webhook deploy endpoint. If not set, endpoint is unauthenticated. + string deploy_secret = 528157198; + // (Write-Only) When true, unsets the deploy secret to allow unauthenticated access to the webhook deploy endpoint. + bool unset_deploy_secret = 535705319; + // The git pull request policy for this project. Valid values are: "off", "links", "recommended", "required". + PullRequestMode pull_request_mode = 432598428; + // Validation policy: If true, the project must pass validation checks before project changes can be committed to the git repository + bool validation_required = 405012041; + // If true, advanced git release management is enabled for this project + bool git_release_mgmt_enabled = 350876702; + // Validation policy: If true, the project can be committed with warnings when `validation_required` is true. (`allow_warnings` does nothing if `validation_required` is false). + bool allow_warnings = 435037044; + // Status of dependencies in your manifest & lockfile + string dependency_status = 440919897; +} + +// Dynamic writeable type for Query removes: + // can, id, slug, share_url, expanded_share_url, url, has_table_calculations +message WriteQuery { + // Model + string model = 102848809; + // Explore Name + string view = 3645563; + // Fields + repeated string fields = 453472492; + // Pivots + repeated string pivots = 446031464; + // Fill Fields + repeated string fill_fields = 520489690; + // Filters + map filters = 487674337; + // Filter Expression + string filter_expression = 175166719; + // Sorting for the query results. Use the format `["view.field", ...]` to sort on fields in ascending order. Use the format `["view.field desc", ...]` to sort on fields in descending order. Use `["__UNSORTED__"]` (2 underscores before and after) to disable sorting entirely. Empty sorts `[]` will trigger a default sort. + repeated string sorts = 109773781; + // Limit + string limit = 110364603; + // Column Limit + string column_limit = 436244100; + // Total + bool total = 102745028; + // Raw Total + string row_total = 379629720; + // Fields on which to run subtotals + repeated string subtotals = 290419819; + // Visualization configuration properties. These properties are typically opaque and differ based on the type of visualization used. There is no specified set of allowed keys. The values can be any type supported by JSON. A "type" key with a string value is often present, and is used by Looker to determine which visualization to present. Visualizations ignore unknown vis_config properties. + map vis_config = 78818913; + // The filter_config represents the state of the filter UI on the explore page for a given query. When running a query via the Looker UI, this parameter takes precedence over "filters". When creating a query or modifying an existing query, "filter_config" should be set to null. Setting it to any other value could cause unexpected filtering behavior. The format should be considered opaque. + map filter_config = 305149514; + // Visible UI Sections + string visible_ui_sections = 364636603; + // Dynamic Fields + string dynamic_fields = 405484086; + // Client Id: used to generate shortened explore URLs. If set by client, must be a unique 22 character alphanumeric string. Otherwise one will be generated. + string client_id = 206142577; + // Query Timezone + string query_timezone = 391786756; +} + +// Dynamic writeable type for RepositoryCredential removes: + // can, id, root_project_id, remote_url, is_configured +message WriteRepositoryCredential { + // Git username for HTTPS authentication. + string git_username = 404090688; + // (Write-Only) Git password for HTTPS authentication. + string git_password = 269826806; + // Public deploy key for SSH authentication. + string ssh_public_key = 525275256; +} + +// Dynamic writeable type for ResultMakerWithIdVisConfigAndDynamicFields removes: + // id, dynamic_fields, filterables, sorts, merge_result_id, total, query_id, sql_query_id, vis_config +message WriteResultMakerWithIdVisConfigAndDynamicFields { + // Dynamic writeable type for Query removes: + // can, id, slug, share_url, expanded_share_url, url, has_table_calculations + WriteQuery query = 115243016; +} + +// Dynamic writeable type for Role removes: + // can, id, url, users_url +message WriteRole { + // Name of Role + string name = 3116757; + // Dynamic writeable type for PermissionSet removes: + // can, all_access, built_in, id, url + WritePermissionSet permission_set = 460987964; + // (Write-Only) Id of permission set + string permission_set_id = 239948280; + // Dynamic writeable type for ModelSet removes: + // can, all_access, built_in, id, url + WriteModelSet model_set = 442444109; + // (Write-Only) Id of model set + string model_set_id = 339578852; +} + +// Dynamic writeable type for SamlConfig removes: + // can, test_slug, modified_at, modified_by, default_new_user_roles, default_new_user_groups, groups, user_attributes, url +message WriteSamlConfig { + // Enable/Disable Saml authentication for the server + bool enabled = 387588912; + // Identity Provider Certificate (provided by IdP) + string idp_cert = 346163572; + // Identity Provider Url (provided by IdP) + string idp_url = 440865893; + // Identity Provider Issuer (provided by IdP) + string idp_issuer = 332335031; + // Identity Provider Audience (set in IdP config). Optional in Looker. Set this only if you want Looker to validate the audience value returned by the IdP. + string idp_audience = 323325369; + // Count of seconds of clock drift to allow when validating timestamps of assertions. + int64 allowed_clock_drift = 535374760; + // Name of user record attributes used to indicate email address field + string user_attribute_map_email = 398964231; + // Name of user record attributes used to indicate first name + string user_attribute_map_first_name = 400740186; + // Name of user record attributes used to indicate last name + string user_attribute_map_last_name = 521517530; + // Merge first-time saml login to existing user account by email addresses. When a user logs in for the first time via saml this option will connect this user into their existing account by finding the account with a matching email address by testing the given types of credentials for existing users. Otherwise a new user account will be created for the user. This list (if provided) must be a comma separated list of string like 'email,ldap,google' + string new_user_migration_types = 331422246; + // Allow alternate email-based login via '/login/email' for admins and for specified users with the 'login_special_email' permission. This option is useful as a fallback during ldap setup, if ldap config problems occur later, or if you need to support some users who are not in your ldap directory. Looker email/password logins are always disabled for regular users when ldap is enabled. + bool alternate_email_login_allowed = 415381803; + // (Write-Only) Array of ids of roles that will be applied to new users the first time they login via Saml + repeated string default_new_user_role_ids = 319694385; + // (Write-Only) Array of ids of groups that will be applied to new users the first time they login via Saml + repeated string default_new_user_group_ids = 417958896; + // Set user roles in Looker based on groups from Saml + bool set_roles_from_groups = 390938262; + // Name of user record attributes used to indicate groups. Used when 'groups_finder_type' is set to 'grouped_attribute_values' + string groups_attribute = 389492843; + // (Read/Write) Array of mappings between Saml Groups and arrays of Looker Role ids + repeated SamlGroupWrite groups_with_role_ids = 475375615; + // Users will not be allowed to login at all unless a role for them is found in Saml if set to true + bool auth_requires_role = 359416088; + // (Read/Write) Array of mappings between Saml User Attributes and arrays of Looker User Attribute ids + repeated SamlUserAttributeWrite user_attributes_with_ids = 332652790; + // Identifier for a strategy for how Looker will find groups in the SAML response. One of ['grouped_attribute_values', 'individual_attributes'] + string groups_finder_type = 21587272; + // Value for group attribute used to indicate membership. Used when 'groups_finder_type' is set to 'individual_attributes' + string groups_member_value = 527664343; + // Bypass the login page when user authentication is required. Redirect to IdP immediately instead. + bool bypass_login_page = 327461774; + // Allow SAML auth'd users to be members of non-reflected Looker groups. If 'false', user will be removed from non-reflected groups on login. + bool allow_normal_group_membership = 215101880; + // SAML auth'd users will inherit roles from non-reflected Looker groups. + bool allow_roles_from_normal_groups = 301477131; + // Allows roles to be directly assigned to SAML auth'd users. + bool allow_direct_roles = 276166815; +} + +// Dynamic writeable type for ScheduledPlan removes: + // id, created_at, updated_at, title, user, next_run_at, last_run_at, can +message WriteScheduledPlan { + // Name of this scheduled plan + string name = 3116757; + // User Id which owns this scheduled plan + string user_id = 413336787; + // Whether schedule is run as recipient (only applicable for email recipients) + bool run_as_recipient = 343878752; + // Whether the ScheduledPlan is enabled + bool enabled = 387588912; + // Id of a look + string look_id = 413287022; + // Id of a dashboard + string dashboard_id = 311679049; + // Id of a LookML dashboard + string lookml_dashboard_id = 351003409; + // Query string to run look or dashboard with + string filters_string = 457599877; + // (DEPRECATED) Alias for filters_string field + string dashboard_filters = 470460328; + // Delivery should occur if running the dashboard or look returns results + bool require_results = 367571767; + // Delivery should occur if the dashboard look does not return results + bool require_no_results = 342203479; + // Delivery should occur if data have changed since the last run + bool require_change = 451184898; + // Will run an unlimited query and send all results. + bool send_all_results = 332106111; + // Vixie-Style crontab specification when to run + string crontab = 331409081; + // Name of a datagroup; if specified will run when datagroup triggered (can't be used with cron string) + string datagroup = 269453013; + // Timezone for interpreting the specified crontab (default is Looker instance timezone) + string timezone = 516241033; + // Query id + string query_id = 279007282; + // Scheduled plan destinations + repeated ScheduledPlanDestination scheduled_plan_destination = 274258130; + // Whether the plan in question should only be run once (usually for testing) + bool run_once = 426696469; + // Whether links back to Looker should be included in this ScheduledPlan + bool include_links = 419825745; + // The size of paper the PDF should be formatted to fit. Valid values are: "letter", "legal", "tabloid", "a0", "a1", "a2", "a3", "a4", "a5". + string pdf_paper_size = 317503911; + // Whether the PDF should be formatted for landscape orientation + bool pdf_landscape = 272762770; + // Whether this schedule is in an embed context or not + bool embed = 95458649; + // Color scheme of the dashboard if applicable + string color_theme = 350454445; + // Whether or not to expand table vis to full length + bool long_tables = 450392146; + // The pixel width at which we render the inline table visualizations + int64 inline_table_width = 380760423; +} + +// Dynamic writeable type for SessionConfig removes: + // can +message WriteSessionConfig { + // Allow users to have persistent sessions when they login + bool allow_persistent_sessions = 399922247; + // Number of minutes for user sessions. Must be between 5 and 43200 + int64 session_minutes = 444808605; + // Allow users to have an unbounded number of concurrent sessions (otherwise, users will be limited to only one session at a time). + bool unlimited_sessions_per_user = 298548380; + // Enforce session logout for sessions that are inactive for 15 minutes. + bool use_inactivity_based_logout = 299884770; + // Track location of session when user logs in. + bool track_session_location = 330330067; +} + +// Dynamic writeable type for Setting +message WriteSetting { + // Toggle extension framework on or off + bool extension_framework_enabled = 350281536; + // Toggle marketplace auto install on or off. Note that auto install only runs if marketplace is enabled. + bool marketplace_auto_install_enabled = 407899176; + // Toggle marketplace on or off + bool marketplace_enabled = 518632083; + // Dynamic writeable type for PrivatelabelConfiguration removes: + // logo_url, favicon_url + WritePrivatelabelConfiguration privatelabel_configuration = 150815432; + CustomWelcomeEmail custom_welcome_email = 199760943; + // Toggle onboarding on or off + bool onboarding_enabled = 456555871; +} + +// Dynamic writeable type for SshServer removes: + // ssh_server_id, finger_print, sha_finger_print, public_key, status +message WriteSshServer { + // The name to identify this SSH Server + string ssh_server_name = 122116528; + // The hostname or ip address of the SSH Server + string ssh_server_host = 422692139; + // The port to connect to on the SSH Server + int64 ssh_server_port = 532989181; + // The username used to connect to the SSH Server + string ssh_server_user = 490745428; +} + +// Dynamic writeable type for SshTunnel removes: + // tunnel_id, ssh_server_name, ssh_server_host, ssh_server_port, ssh_server_user, last_attempt, local_host_port, status +message WriteSshTunnel { + // SSH Server ID + string ssh_server_id = 20333920; + // Hostname or IP Address of the Database Server + string database_host = 495546218; + // Port that the Database Server is listening on + int64 database_port = 400671025; +} + +// Dynamic writeable type for Theme removes: + // can, id +message WriteTheme { + // Timestamp for when this theme becomes active. Null=always + google.protobuf.Timestamp begin_at = 489666644; + // Timestamp for when this theme expires. Null=never + google.protobuf.Timestamp end_at = 440737243; + // Name of theme. Can only be alphanumeric and underscores. + string name = 3116757; + ThemeSettings settings = 316732240; +} + +// Dynamic writeable type for User removes: + // can, avatar_url, avatar_url_without_sizing, credentials_api3, credentials_embed, credentials_google, credentials_ldap, credentials_looker_openid, credentials_oidc, credentials_saml, credentials_totp, display_name, email, embed_group_space_id, group_ids, id, looker_versions, personal_folder_id, presumed_looker_employee, role_ids, sessions, verified_looker_employee, roles_externally_managed, allow_direct_roles, allow_normal_group_membership, allow_roles_from_normal_groups, embed_group_folder_id, url +message WriteUser { + // Dynamic writeable type for CredentialsEmail removes: + // can, created_at, is_disabled, logged_in_at, password_reset_url, type, url, user_url + WriteCredentialsEmail credentials_email = 381333772; + // First name + string first_name = 277109009; + // ID string for user's home folder + string home_folder_id = 115041580; + // Account has been disabled + bool is_disabled = 332107427; + // Last name + string last_name = 510613627; + // User's preferred locale. User locale takes precedence over Looker's system-wide default locale. Locale determines language of display strings and date and numeric formatting in API responses. Locale string must be a 2 letter language code or a combination of language code and region code: 'en' or 'en-US', for example. + string locale = 325173590; + // User's dev workspace has been checked for presence of applicable production projects + bool models_dir_validated = 303600374; + // Per user dictionary of undocumented state information owned by the Looker UI. + map ui_state = 401473497; +} + +// Dynamic writeable type for UserAttribute removes: + // can, id, is_system, is_permanent +message WriteUserAttribute { + // Name of user attribute + string name = 3116757; + // Human-friendly label for user attribute + string label = 102846452; + // Type of user attribute ("string", "number", "datetime", "yesno", "zipcode") + string type = 3120390; + // Default value for when no value is set on the user + string default_value = 429771091; + // If true, users will not be able to view values of this attribute + bool value_is_hidden = 527294216; + // Non-admin users can see the values of their attributes and use them in filters + bool user_can_view = 177700632; + // Users can change the value of this attribute for themselves + bool user_can_edit = 329711878; + // Destinations to which a hidden attribute may be sent. Once set, cannot be edited. + string hidden_value_domain_whitelist = 342844307; +} + +// Dynamic writeable type for UserAttributeWithValue removes: + // can, name, label, rank, user_id, user_can_edit, value_is_hidden, user_attribute_id, source, hidden_value_domain_whitelist +message WriteUserAttributeWithValue { + // Value of attribute for user + string value = 96868081; +} + +// Dynamic writeable type for WhitelabelConfiguration removes: + // id, logo_url, favicon_url +message WriteWhitelabelConfiguration { + // Customer logo image. Expected base64 encoded data (write-only) + string logo_file = 302348988; + // Custom favicon image. Expected base64 encoded data (write-only) + string favicon_file = 348157260; + // Default page title + string default_title = 491557590; + // Boolean to toggle showing help menus + bool show_help_menu = 319857861; + // Boolean to toggle showing docs + bool show_docs = 430915951; + // Boolean to toggle showing email subscription options. + bool show_email_sub_options = 481532061; + // Boolean to toggle mentions of Looker in emails + bool allow_looker_mentions = 373852627; + // Boolean to toggle links to Looker in emails + bool allow_looker_links = 375199970; + // Allow subject line and email heading customization in customized emails” + bool custom_welcome_email_advanced = 119812016; + // Remove the word Looker from appearing in the account setup page + bool setup_mentions = 338894507; + // Remove Looker logo from Alerts + bool alerts_logo = 409573540; + // Remove Looker links from Alerts + bool alerts_links = 304998460; + // Remove Looker mentions in home folder page when you don’t have any items saved + bool folders_mentions = 496168113; +} + +message SearchAlertsReq { + // (Optional) Number of results to return (used with `offset`). + int64 limit = 110364603; + // (Optional) Number of results to skip before returning any (used with `limit`). + int64 offset = 438591449; + // (Optional) Dimension by which to order the results(`dashboard` | `owner`) + string group_by = 34500105; + // (Optional) Requested fields. + string fields = 453472492; + // (Optional) Filter on returning only enabled or disabled alerts. + bool disabled = 408194946; + // (Optional) Filter on alert frequency, such as: monthly, weekly, daily, hourly, minutes + string frequency = 342468377; + // (Optional) Filter on whether the alert has met its condition when it last executed + bool condition_met = 433973774; + // (Optional) Filter on the start range of the last time the alerts were run. Example: 2021-01-01T01:01:01-08:00. + string last_run_start = 315096022; + // (Optional) Filter on the start range of the last time the alerts were run. Example: 2021-01-01T01:01:01-08:00. + string last_run_end = 100909630; + // (Admin only) (Optional) Filter for all owners. + bool all_owners = 317015952; +} + +message SearchAlertsResponse { + // Alert. + repeated Alert result = 1; +} + +message SearchAlertsStreamResponse { + // Alert. + Alert result = 1; +} + +message GetAlertReq { + // ID of an alert + string alert_id = 283625182; +} + +message GetAlertResponse { + // Alert + Alert result = 1; +} + +message UpdateAlertReq { + // ID of an alert + string alert_id = 283625182; + Alert body = 3704350; +} + +message UpdateAlertResponse { + // The alert is saved. + Alert result = 1; +} + +message UpdateAlertFieldReq { + // ID of an alert + string alert_id = 283625182; + AlertPatch body = 3704350; +} + +message UpdateAlertFieldResponse { + // The alert is saved. + Alert result = 1; +} + +message DeleteAlertReq { + // ID of an alert + string alert_id = 283625182; +} + +message DeleteAlertResponse { + // Alert successfully deleted. +} + +message CreateAlertReq { + Alert body = 3704350; +} + +message CreateAlertResponse { + // The alert is saved. + Alert result = 1; +} + +message EnqueueAlertReq { + // ID of an alert + string alert_id = 283625182; + // Whether to enqueue an alert again if its already running. + bool force = 96338027; +} + +message EnqueueAlertResponse { + // Alert successfully added to the queue. Does not indicate it has been run +} + +message LoginReq { + // client_id part of API3 Key. + string client_id = 206142577; + // client_secret part of API3 Key. + string client_secret = 509513545; +} + +message LoginResponse { + // Access token with metadata. + AccessToken result = 1; +} + +message LoginUserReq { + // Id of user. + string user_id = 413336787; + // When true (default), API calls using the returned access_token are attributed to the admin user who created the access_token. When false, API activity is attributed to the user the access_token runs as. False requires a looker license. + bool associative = 29418031; +} + +message LoginUserResponse { + // Access token with metadata. + AccessToken result = 1; +} + +message LogoutReq { + +} + +message LogoutResponse { + // Logged out successfully. + string result = 1; +} + +message CreateEmbedSecretReq { + EmbedSecret body = 3704350; +} + +message CreateEmbedSecretResponse { + // embed secret + EmbedSecret result = 1; +} + +message DeleteEmbedSecretReq { + // Id of Embed Secret + string embed_secret_id = 21080676; +} + +message DeleteEmbedSecretResponse { + // Successfully deleted. + string result = 1; +} + +message CreateSsoEmbedUrlReq { + EmbedSsoParams body = 3704350; +} + +message CreateSsoEmbedUrlResponse { + // Signed SSO URL + EmbedUrlResponse result = 1; +} + +message CreateEmbedUrlAsMeReq { + EmbedParams body = 3704350; +} + +message CreateEmbedUrlAsMeResponse { + // Embed URL + EmbedUrlResponse result = 1; +} + +message LdapConfigReq { + +} + +message LdapConfigResponse { + // LDAP Configuration. + LDAPConfig result = 1; +} + +message UpdateLdapConfigReq { + LDAPConfig body = 3704350; +} + +message UpdateLdapConfigResponse { + // New state for LDAP Configuration. + LDAPConfig result = 1; +} + +message TestLdapConfigConnectionReq { + LDAPConfig body = 3704350; +} + +message TestLdapConfigConnectionResponse { + // Result info. + LDAPConfigTestResult result = 1; +} + +message TestLdapConfigAuthReq { + LDAPConfig body = 3704350; +} + +message TestLdapConfigAuthResponse { + // Result info. + LDAPConfigTestResult result = 1; +} + +message TestLdapConfigUserInfoReq { + LDAPConfig body = 3704350; +} + +message TestLdapConfigUserInfoResponse { + // Result info. + LDAPConfigTestResult result = 1; +} + +message TestLdapConfigUserAuthReq { + LDAPConfig body = 3704350; +} + +message TestLdapConfigUserAuthResponse { + // Result info. + LDAPConfigTestResult result = 1; +} + +message AllOauthClientAppsReq { + // Requested fields. + string fields = 453472492; +} + +message AllOauthClientAppsResponse { + // OAuth Client App + repeated OauthClientApp result = 1; +} + +message AllOauthClientAppsStreamResponse { + // OAuth Client App + OauthClientApp result = 1; +} + +message OauthClientAppReq { + // The unique id of this application + string client_guid = 368914559; + // Requested fields. + string fields = 453472492; +} + +message OauthClientAppResponse { + // OAuth Client App + OauthClientApp result = 1; +} + +message RegisterOauthClientAppReq { + // The unique id of this application + string client_guid = 368914559; + OauthClientApp body = 3704350; + // Requested fields. + string fields = 453472492; +} + +message RegisterOauthClientAppResponse { + // OAuth Client App + OauthClientApp result = 1; +} + +message UpdateOauthClientAppReq { + // The unique id of this application + string client_guid = 368914559; + OauthClientApp body = 3704350; + // Requested fields. + string fields = 453472492; +} + +message UpdateOauthClientAppResponse { + // OAuth Client App + OauthClientApp result = 1; +} + +message DeleteOauthClientAppReq { + // The unique id of this application + string client_guid = 368914559; +} + +message DeleteOauthClientAppResponse { + // Successfully deleted. + string result = 1; +} + +message InvalidateTokensReq { + // The unique id of the application + string client_guid = 368914559; +} + +message InvalidateTokensResponse { + // Successfully deleted. + string result = 1; +} + +message ActivateAppUserReq { + // The unique id of this application + string client_guid = 368914559; + // The id of the user to enable use of this app + string user_id = 413336787; + // Requested fields. + string fields = 453472492; +} + +message ActivateAppUserResponse { + +} + +message DeactivateAppUserReq { + // The unique id of this application + string client_guid = 368914559; + // The id of the user to enable use of this app + string user_id = 413336787; + // Requested fields. + string fields = 453472492; +} + +message DeactivateAppUserResponse { + // Successfully deleted. + string result = 1; +} + +message OidcConfigReq { + +} + +message OidcConfigResponse { + // OIDC Configuration. + OIDCConfig result = 1; +} + +message UpdateOidcConfigReq { + OIDCConfig body = 3704350; +} + +message UpdateOidcConfigResponse { + // New state for OIDC Configuration. + OIDCConfig result = 1; +} + +message OidcTestConfigReq { + // Slug of test config + string test_slug = 505468220; +} + +message OidcTestConfigResponse { + // OIDC test config. + OIDCConfig result = 1; +} + +message DeleteOidcTestConfigReq { + // Slug of test config + string test_slug = 505468220; +} + +message DeleteOidcTestConfigResponse { + // Test config succssfully deleted. + string result = 1; +} + +message CreateOidcTestConfigReq { + OIDCConfig body = 3704350; +} + +message CreateOidcTestConfigResponse { + // OIDC test config + OIDCConfig result = 1; +} + +message PasswordConfigReq { + +} + +message PasswordConfigResponse { + // Password Config + PasswordConfig result = 1; +} + +message UpdatePasswordConfigReq { + PasswordConfig body = 3704350; +} + +message UpdatePasswordConfigResponse { + // Password Config + PasswordConfig result = 1; +} + +message ForcePasswordResetAtNextLoginForAllUsersReq { + +} + +message ForcePasswordResetAtNextLoginForAllUsersResponse { + // Password Config + string result = 1; +} + +message SamlConfigReq { + +} + +message SamlConfigResponse { + // SAML Configuration. + SamlConfig result = 1; +} + +message UpdateSamlConfigReq { + SamlConfig body = 3704350; +} + +message UpdateSamlConfigResponse { + // New state for SAML Configuration. + SamlConfig result = 1; +} + +message SamlTestConfigReq { + // Slug of test config + string test_slug = 505468220; +} + +message SamlTestConfigResponse { + // SAML test config. + SamlConfig result = 1; +} + +message DeleteSamlTestConfigReq { + // Slug of test config + string test_slug = 505468220; +} + +message DeleteSamlTestConfigResponse { + // Test config succssfully deleted. + string result = 1; +} + +message CreateSamlTestConfigReq { + SamlConfig body = 3704350; +} + +message CreateSamlTestConfigResponse { + // SAML test config + SamlConfig result = 1; +} + +message ParseSamlIdpMetadataReq { + string body = 3704350; +} + +message ParseSamlIdpMetadataResponse { + // Parse result + SamlMetadataParseResult result = 1; +} + +message FetchAndParseSamlIdpMetadataReq { + string body = 3704350; +} + +message FetchAndParseSamlIdpMetadataResponse { + // Parse result + SamlMetadataParseResult result = 1; +} + +message SessionConfigReq { + +} + +message SessionConfigResponse { + // Session Config + SessionConfig result = 1; +} + +message UpdateSessionConfigReq { + SessionConfig body = 3704350; +} + +message UpdateSessionConfigResponse { + // Session Config + SessionConfig result = 1; +} + +message GetSupportAccessAllowlistEntriesReq { + // Requested fields. + string fields = 453472492; +} + +message GetSupportAccessAllowlistEntriesResponse { + // Support Access Allowlist Entries + repeated SupportAccessAllowlistEntry result = 1; +} + +message GetSupportAccessAllowlistEntriesStreamResponse { + // Support Access Allowlist Entries + SupportAccessAllowlistEntry result = 1; +} + +message AddSupportAccessAllowlistEntriesReq { + SupportAccessAddEntries body = 3704350; +} + +message AddSupportAccessAllowlistEntriesResponse { + // Support Access Allowlist Entries + repeated SupportAccessAllowlistEntry result = 1; +} + +message AddSupportAccessAllowlistEntriesStreamResponse { + // Support Access Allowlist Entries + SupportAccessAllowlistEntry result = 1; +} + +message DeleteSupportAccessAllowlistEntryReq { + // Id of Allowlist Entry + string entry_id = 278993096; +} + +message DeleteSupportAccessAllowlistEntryResponse { + // Entry successfully deleted. + string result = 1; +} + +message EnableSupportAccessReq { + SupportAccessEnable body = 3704350; +} + +message EnableSupportAccessResponse { + // Support Access Status + SupportAccessStatus result = 1; +} + +message DisableSupportAccessReq { + +} + +message DisableSupportAccessResponse { + // Support Access Status + SupportAccessStatus result = 1; +} + +message SupportAccessStatusReq { + +} + +message SupportAccessStatusResponse { + // Support Access Status + SupportAccessStatus result = 1; +} + +message AllUserLoginLockoutsReq { + // Include only these fields in the response + string fields = 453472492; +} + +message AllUserLoginLockoutsResponse { + // User Login Lockout + repeated UserLoginLockout result = 1; +} + +message AllUserLoginLockoutsStreamResponse { + // User Login Lockout + UserLoginLockout result = 1; +} + +message SearchUserLoginLockoutsReq { + // Include only these fields in the response + string fields = 453472492; + // DEPRECATED. Use limit and offset instead. Return only page N of paginated results + int64 page = 3110993; + // DEPRECATED. Use limit and offset instead. Return N rows of data per page + int64 per_page = 262811729; + // Number of results to return. (used with offset and takes priority over page and per_page) + int64 limit = 110364603; + // Number of results to skip before returning any. (used with limit and takes priority over page and per_page) + int64 offset = 438591449; + // Fields to sort by. + string sorts = 109773781; + // Auth type user is locked out for (email, ldap, totp, api) + string auth_type = 278079987; + // Match name + string full_name = 510674737; + // Match email + string email = 102965020; + // Match remote LDAP ID + string remote_id = 314868582; + // Combine given search criteria in a boolean OR expression + bool filter_or = 515276450; +} + +message SearchUserLoginLockoutsResponse { + // User Login Lockout + repeated UserLoginLockout result = 1; +} + +message SearchUserLoginLockoutsStreamResponse { + // User Login Lockout + UserLoginLockout result = 1; +} + +message DeleteUserLoginLockoutReq { + // The key associated with the locked user + string key = 119519; +} + +message DeleteUserLoginLockoutResponse { + // Successfully deleted. + string result = 1; +} + +message AllBoardsReq { + // Requested fields. + string fields = 453472492; +} + +message AllBoardsResponse { + // Board + repeated Board result = 1; +} + +message AllBoardsStreamResponse { + // Board + Board result = 1; +} + +message CreateBoardReq { + Board body = 3704350; + // Requested fields. + string fields = 453472492; +} + +message CreateBoardResponse { + // Board + Board result = 1; +} + +message SearchBoardsReq { + // Matches board title. + string title = 96607896; + // Matches the timestamp for when the board was created. + string created_at = 464589301; + // The first name of the user who created this board. + string first_name = 277109009; + // The last name of the user who created this board. + string last_name = 510613627; + // Requested fields. + string fields = 453472492; + // Return favorited boards when true. + bool favorited = 433906254; + // Filter on boards created by a particular user. + string creator_id = 466744818; + // The fields to sort the results by + string sorts = 109773781; + // The page to return. DEPRECATED. Use offset instead. + int64 page = 3110993; + // The number of items in the returned page. DEPRECATED. Use limit instead. + int64 per_page = 262811729; + // The number of items to skip before returning any. (used with limit and takes priority over page and per_page) + int64 offset = 438591449; + // The maximum number of items to return. (used with offset and takes priority over page and per_page) + int64 limit = 110364603; + // Combine given search criteria in a boolean OR expression + bool filter_or = 515276450; +} + +message SearchBoardsResponse { + // boards + repeated Board result = 1; +} + +message SearchBoardsStreamResponse { + // boards + Board result = 1; +} + +message BoardReq { + // Id of board + string board_id = 298405268; + // Requested fields. + string fields = 453472492; +} + +message BoardResponse { + // Board + Board result = 1; +} + +message UpdateBoardReq { + // Id of board + string board_id = 298405268; + Board body = 3704350; + // Requested fields. + string fields = 453472492; +} + +message UpdateBoardResponse { + // Board + Board result = 1; +} + +message DeleteBoardReq { + // Id of board + string board_id = 298405268; +} + +message DeleteBoardResponse { + // Successfully deleted. + string result = 1; +} + +message AllBoardItemsReq { + // Requested fields. + string fields = 453472492; + // Fields to sort by. + string sorts = 109773781; + // Filter to a specific board section + string board_section_id = 473711058; +} + +message AllBoardItemsResponse { + // Board Item + repeated BoardItem result = 1; +} + +message AllBoardItemsStreamResponse { + // Board Item + BoardItem result = 1; +} + +message CreateBoardItemReq { + BoardItem body = 3704350; + // Requested fields. + string fields = 453472492; +} + +message CreateBoardItemResponse { + // Board Item + BoardItem result = 1; +} + +message BoardItemReq { + // Id of board item + string board_item_id = 526513540; + // Requested fields. + string fields = 453472492; +} + +message BoardItemResponse { + // Board Item + BoardItem result = 1; +} + +message UpdateBoardItemReq { + // Id of board item + string board_item_id = 526513540; + BoardItem body = 3704350; + // Requested fields. + string fields = 453472492; +} + +message UpdateBoardItemResponse { + // Board Item + BoardItem result = 1; +} + +message DeleteBoardItemReq { + // Id of board item + string board_item_id = 526513540; +} + +message DeleteBoardItemResponse { + // Successfully deleted. + string result = 1; +} + +message AllBoardSectionsReq { + // Requested fields. + string fields = 453472492; + // Fields to sort by. + string sorts = 109773781; +} + +message AllBoardSectionsResponse { + // Board section + repeated BoardSection result = 1; +} + +message AllBoardSectionsStreamResponse { + // Board section + BoardSection result = 1; +} + +message CreateBoardSectionReq { + BoardSection body = 3704350; + // Requested fields. + string fields = 453472492; +} + +message CreateBoardSectionResponse { + // Board section + BoardSection result = 1; +} + +message BoardSectionReq { + // Id of board section + string board_section_id = 473711058; + // Requested fields. + string fields = 453472492; +} + +message BoardSectionResponse { + // Board section + BoardSection result = 1; +} + +message UpdateBoardSectionReq { + // Id of board section + string board_section_id = 473711058; + BoardSection body = 3704350; + // Requested fields. + string fields = 453472492; +} + +message UpdateBoardSectionResponse { + // Board section + BoardSection result = 1; +} + +message DeleteBoardSectionReq { + // Id of board section + string board_section_id = 473711058; +} + +message DeleteBoardSectionResponse { + // Successfully deleted. + string result = 1; +} + +message AllColorCollectionsReq { + // Requested fields. + string fields = 453472492; +} + +message AllColorCollectionsResponse { + // ColorCollections + repeated ColorCollection result = 1; +} + +message AllColorCollectionsStreamResponse { + // ColorCollections + ColorCollection result = 1; +} + +message CreateColorCollectionReq { + ColorCollection body = 3704350; +} + +message CreateColorCollectionResponse { + // ColorCollection + ColorCollection result = 1; +} + +message ColorCollectionsCustomReq { + // Requested fields. + string fields = 453472492; +} + +message ColorCollectionsCustomResponse { + // ColorCollections + repeated ColorCollection result = 1; +} + +message ColorCollectionsCustomStreamResponse { + // ColorCollections + ColorCollection result = 1; +} + +message ColorCollectionsStandardReq { + // Requested fields. + string fields = 453472492; +} + +message ColorCollectionsStandardResponse { + // ColorCollections + repeated ColorCollection result = 1; +} + +message ColorCollectionsStandardStreamResponse { + // ColorCollections + ColorCollection result = 1; +} + +message DefaultColorCollectionReq { + +} + +message DefaultColorCollectionResponse { + // ColorCollection + ColorCollection result = 1; +} + +message SetDefaultColorCollectionReq { + // ID of color collection to set as default + string collection_id = 394557532; +} + +message SetDefaultColorCollectionResponse { + // ColorCollection + ColorCollection result = 1; +} + +message ColorCollectionReq { + // Id of Color Collection + string collection_id = 394557532; + // Requested fields. + string fields = 453472492; +} + +message ColorCollectionResponse { + // ColorCollection + ColorCollection result = 1; +} + +message UpdateColorCollectionReq { + // Id of Custom Color Collection + string collection_id = 394557532; + ColorCollection body = 3704350; +} + +message UpdateColorCollectionResponse { + // ColorCollection + ColorCollection result = 1; +} + +message DeleteColorCollectionReq { + // Id of Color Collection + string collection_id = 394557532; +} + +message DeleteColorCollectionResponse { + // Successfully deleted. + string result = 1; +} + +message CloudStorageConfigurationReq { + +} + +message CloudStorageConfigurationResponse { + // Current Cloud Storage Configuration + BackupConfiguration result = 1; +} + +message UpdateCloudStorageConfigurationReq { + BackupConfiguration body = 3704350; +} + +message UpdateCloudStorageConfigurationResponse { + // New state for specified model set. + BackupConfiguration result = 1; +} + +message CustomWelcomeEmailReq { + +} + +message CustomWelcomeEmailResponse { + // Custom Welcome Email + CustomWelcomeEmail result = 1; +} + +message UpdateCustomWelcomeEmailReq { + CustomWelcomeEmail body = 3704350; + // If true a test email with the content from the request will be sent to the current user after saving + bool send_test_welcome_email = 371686228; +} + +message UpdateCustomWelcomeEmailResponse { + // Custom Welcome Email + CustomWelcomeEmail result = 1; +} + +message UpdateCustomWelcomeEmailTestReq { + WelcomeEmailTest body = 3704350; +} + +message UpdateCustomWelcomeEmailTestResponse { + // Send Test Welcome Email + WelcomeEmailTest result = 1; +} + +message DigestEmailsEnabledReq { + +} + +message DigestEmailsEnabledResponse { + // Digest_emails + DigestEmails result = 1; +} + +message UpdateDigestEmailsEnabledReq { + DigestEmails body = 3704350; +} + +message UpdateDigestEmailsEnabledResponse { + // Digest_emails + DigestEmails result = 1; +} + +message CreateDigestEmailSendReq { + +} + +message CreateDigestEmailSendResponse { + // Status of generating and sending the data + DigestEmailSend result = 1; +} + +message PublicEgressIpAddressesReq { + +} + +message PublicEgressIpAddressesResponse { + // Public Egress IP Addresses + EgressIpAddresses result = 1; +} + +message InternalHelpResourcesContentReq { + +} + +message InternalHelpResourcesContentResponse { + // Internal Help Resources Content + InternalHelpResourcesContent result = 1; +} + +message UpdateInternalHelpResourcesContentReq { + InternalHelpResourcesContent body = 3704350; +} + +message UpdateInternalHelpResourcesContentResponse { + // Internal Help Resources Content + InternalHelpResourcesContent result = 1; +} + +message InternalHelpResourcesReq { + +} + +message InternalHelpResourcesResponse { + // Internal Help Resources + InternalHelpResources result = 1; +} + +message UpdateInternalHelpResourcesReq { + InternalHelpResources body = 3704350; +} + +message UpdateInternalHelpResourcesResponse { + // Custom Welcome Email + InternalHelpResources result = 1; +} + +message AllLegacyFeaturesReq { + +} + +message AllLegacyFeaturesResponse { + // Legacy Feature + repeated LegacyFeature result = 1; +} + +message AllLegacyFeaturesStreamResponse { + // Legacy Feature + LegacyFeature result = 1; +} + +message LegacyFeatureReq { + // id of legacy feature + string legacy_feature_id = 443075361; +} + +message LegacyFeatureResponse { + // Legacy Feature + LegacyFeature result = 1; +} + +message UpdateLegacyFeatureReq { + // id of legacy feature + string legacy_feature_id = 443075361; + LegacyFeature body = 3704350; +} + +message UpdateLegacyFeatureResponse { + // Legacy Feature + LegacyFeature result = 1; +} + +message AllLocalesReq { + +} + +message AllLocalesResponse { + // Locale + repeated Locale result = 1; +} + +message AllLocalesStreamResponse { + // Locale + Locale result = 1; +} + +message MobileSettingsReq { + +} + +message MobileSettingsResponse { + // Mobile_Settings + MobileSettings result = 1; +} + +message GetSettingReq { + // Requested fields + string fields = 453472492; +} + +message GetSettingResponse { + // Looker Settings + Setting result = 1; +} + +message SetSettingReq { + Setting body = 3704350; + // Requested fields + string fields = 453472492; +} + +message SetSettingResponse { + // Looker Settings + Setting result = 1; +} + +message SetSmtpSettingsReq { + SmtpSettings body = 3704350; +} + +message SetSmtpSettingsResponse { + // Successfully updated SMTP settings +} + +message SmtpStatusReq { + // Include only these fields in the response + string fields = 453472492; +} + +message SmtpStatusResponse { + // SMTP Status + SmtpStatus result = 1; +} + +message AllTimezonesReq { + +} + +message AllTimezonesResponse { + // Timezone + repeated Timezone result = 1; +} + +message AllTimezonesStreamResponse { + // Timezone + Timezone result = 1; +} + +message VersionsReq { + // Requested fields. + string fields = 453472492; +} + +message VersionsResponse { + // ApiVersion + ApiVersion result = 1; +} + +message ApiSpecReq { + // API version + string api_version = 394772387; + // Specification name. Typically, this is "swagger.json" + string specification = 360141264; +} + +message ApiSpecResponse { + // API specification + google.protobuf.Any result = 1; +} + +message WhitelabelConfigurationReq { + // Requested fields. + string fields = 453472492; +} + +message WhitelabelConfigurationResponse { + // Whitelabel configuration + WhitelabelConfiguration result = 1; +} + +message UpdateWhitelabelConfigurationReq { + WhitelabelConfiguration body = 3704350; +} + +message UpdateWhitelabelConfigurationResponse { + // Whitelabel configuration + WhitelabelConfiguration result = 1; +} + +message AllConnectionsReq { + // Requested fields. + string fields = 453472492; +} + +message AllConnectionsResponse { + // Connection + repeated DBConnection result = 1; +} + +message AllConnectionsStreamResponse { + // Connection + DBConnection result = 1; +} + +message CreateConnectionReq { + DBConnection body = 3704350; +} + +message CreateConnectionResponse { + // Connection + DBConnection result = 1; +} + +message ConnectionReq { + // Name of connection + string connection_name = 292836428; + // Requested fields. + string fields = 453472492; +} + +message ConnectionResponse { + // Connection + DBConnection result = 1; +} + +message UpdateConnectionReq { + // Name of connection + string connection_name = 292836428; + DBConnection body = 3704350; +} + +message UpdateConnectionResponse { + // Connection + DBConnection result = 1; +} + +message DeleteConnectionReq { + // Name of connection + string connection_name = 292836428; +} + +message DeleteConnectionResponse { + // Successfully deleted. + string result = 1; +} + +message DeleteConnectionOverrideReq { + // Name of connection + string connection_name = 292836428; + // Context of connection override + string override_context = 533032546; +} + +message DeleteConnectionOverrideResponse { + // Successfully deleted. + string result = 1; +} + +message TestConnectionReq { + // Name of connection + string connection_name = 292836428; + // Array of names of tests to run + string tests = 109774433; +} + +message TestConnectionResponse { + // Test results + repeated DBConnectionTestResult result = 1; +} + +message TestConnectionStreamResponse { + // Test results + DBConnectionTestResult result = 1; +} + +message TestConnectionConfigReq { + DBConnection body = 3704350; + // Array of names of tests to run + string tests = 109774433; +} + +message TestConnectionConfigResponse { + // Test results + repeated DBConnectionTestResult result = 1; +} + +message TestConnectionConfigStreamResponse { + // Test results + DBConnectionTestResult result = 1; +} + +message AllDialectInfosReq { + // Requested fields. + string fields = 453472492; +} + +message AllDialectInfosResponse { + // Dialect Info + repeated DialectInfo result = 1; +} + +message AllDialectInfosStreamResponse { + // Dialect Info + DialectInfo result = 1; +} + +message AllExternalOauthApplicationsReq { + // Application name + string name = 3116757; + // Application Client ID + string client_id = 206142577; +} + +message AllExternalOauthApplicationsResponse { + // External OAuth Application + repeated ExternalOauthApplication result = 1; +} + +message AllExternalOauthApplicationsStreamResponse { + // External OAuth Application + ExternalOauthApplication result = 1; +} + +message CreateExternalOauthApplicationReq { + ExternalOauthApplication body = 3704350; +} + +message CreateExternalOauthApplicationResponse { + // External OAuth Application + ExternalOauthApplication result = 1; +} + +message CreateOauthApplicationUserStateReq { + CreateOAuthApplicationUserStateRequest body = 3704350; +} + +message CreateOauthApplicationUserStateResponse { + // Created user state + CreateOAuthApplicationUserStateResponse result = 1; +} + +message AllSshServersReq { + // Requested fields. + string fields = 453472492; +} + +message AllSshServersResponse { + // SSH Server + repeated SshServer result = 1; +} + +message AllSshServersStreamResponse { + // SSH Server + SshServer result = 1; +} + +message CreateSshServerReq { + SshServer body = 3704350; +} + +message CreateSshServerResponse { + // SSH Server + SshServer result = 1; +} + +message SshServerReq { + // Id of SSH Server + string ssh_server_id = 20333920; +} + +message SshServerResponse { + // SSH Server + SshServer result = 1; +} + +message UpdateSshServerReq { + // Id of SSH Server + string ssh_server_id = 20333920; + SshServer body = 3704350; +} + +message UpdateSshServerResponse { + // SSH Server + SshServer result = 1; +} + +message DeleteSshServerReq { + // Id of SSH Server + string ssh_server_id = 20333920; +} + +message DeleteSshServerResponse { + // Successfully deleted. + string result = 1; +} + +message TestSshServerReq { + // Id of SSH Server + string ssh_server_id = 20333920; +} + +message TestSshServerResponse { + // Test SSH Server + SshServer result = 1; +} + +message AllSshTunnelsReq { + // Requested fields. + string fields = 453472492; +} + +message AllSshTunnelsResponse { + // SSH Tunnel + repeated SshTunnel result = 1; +} + +message AllSshTunnelsStreamResponse { + // SSH Tunnel + SshTunnel result = 1; +} + +message CreateSshTunnelReq { + SshTunnel body = 3704350; +} + +message CreateSshTunnelResponse { + // SSH Tunnel + SshTunnel result = 1; +} + +message SshTunnelReq { + // Id of SSH Tunnel + string ssh_tunnel_id = 326277206; +} + +message SshTunnelResponse { + // SSH Tunnel + SshTunnel result = 1; +} + +message UpdateSshTunnelReq { + // Id of SSH Tunnel + string ssh_tunnel_id = 326277206; + SshTunnel body = 3704350; +} + +message UpdateSshTunnelResponse { + // SSH Tunnel + SshTunnel result = 1; +} + +message DeleteSshTunnelReq { + // Id of SSH Tunnel + string ssh_tunnel_id = 326277206; +} + +message DeleteSshTunnelResponse { + // Successfully deleted. + string result = 1; +} + +message TestSshTunnelReq { + // Id of SSH Tunnel + string ssh_tunnel_id = 326277206; +} + +message TestSshTunnelResponse { + // Test SSH Tunnel + SshTunnel result = 1; +} + +message SshPublicKeyReq { + +} + +message SshPublicKeyResponse { + // SSH Public Key + SshPublicKey result = 1; +} + +message SearchContentFavoritesReq { + // Match content favorite id(s) + string id = 3205; + // Match user id(s).To create a list of multiple ids, use commas as separators + string user_id = 413336787; + // Match content metadata id(s).To create a list of multiple ids, use commas as separators + string content_metadata_id = 4111259; + // Match dashboard id(s).To create a list of multiple ids, use commas as separators + string dashboard_id = 311679049; + // Match look id(s).To create a list of multiple ids, use commas as separators + string look_id = 413287022; + // Match board id(s).To create a list of multiple ids, use commas as separators + string board_id = 298405268; + // Number of results to return. (used with offset) + int64 limit = 110364603; + // Number of results to skip before returning any. (used with limit) + int64 offset = 438591449; + // Fields to sort by. + string sorts = 109773781; + // Requested fields. + string fields = 453472492; + // Combine given search criteria in a boolean OR expression + bool filter_or = 515276450; +} + +message SearchContentFavoritesResponse { + // Favorite Content + repeated ContentFavorite result = 1; +} + +message SearchContentFavoritesStreamResponse { + // Favorite Content + ContentFavorite result = 1; +} + +message ContentFavoriteReq { + // Id of favorite content + string content_favorite_id = 363282886; + // Requested fields. + string fields = 453472492; +} + +message ContentFavoriteResponse { + // Favorite Content + ContentFavorite result = 1; +} + +message DeleteContentFavoriteReq { + // Id of favorite content + string content_favorite_id = 363282886; +} + +message DeleteContentFavoriteResponse { + // Successfully deleted. + string result = 1; +} + +message CreateContentFavoriteReq { + ContentFavorite body = 3704350; +} + +message CreateContentFavoriteResponse { + // Favorite Content + ContentFavorite result = 1; +} + +message AllContentMetadatasReq { + // Parent space of content. + string parent_id = 206134256; + // Requested fields. + string fields = 453472492; +} + +message AllContentMetadatasResponse { + // Content Metadata + repeated ContentMeta result = 1; +} + +message AllContentMetadatasStreamResponse { + // Content Metadata + ContentMeta result = 1; +} + +message ContentMetadataReq { + // Id of content metadata + string content_metadata_id = 4111259; + // Requested fields. + string fields = 453472492; +} + +message ContentMetadataResponse { + // Content Metadata + ContentMeta result = 1; +} + +message UpdateContentMetadataReq { + // Id of content metadata + string content_metadata_id = 4111259; + ContentMeta body = 3704350; +} + +message UpdateContentMetadataResponse { + // Content Metadata + ContentMeta result = 1; +} + +message AllContentMetadataAccessesReq { + // Id of content metadata + string content_metadata_id = 4111259; + // Requested fields. + string fields = 453472492; +} + +message AllContentMetadataAccessesResponse { + // Content Metadata Access + repeated ContentMetaGroupUser result = 1; +} + +message AllContentMetadataAccessesStreamResponse { + // Content Metadata Access + ContentMetaGroupUser result = 1; +} + +message CreateContentMetadataAccessReq { + ContentMetaGroupUser body = 3704350; + // Optionally sends notification email when granting access to a board. + bool send_boards_notification_email = 496802787; +} + +message CreateContentMetadataAccessResponse { + // Content Metadata Access + ContentMetaGroupUser result = 1; +} + +message UpdateContentMetadataAccessReq { + // Id of content metadata access + string content_metadata_access_id = 393434598; + ContentMetaGroupUser body = 3704350; +} + +message UpdateContentMetadataAccessResponse { + // Content Metadata Access + ContentMetaGroupUser result = 1; +} + +message DeleteContentMetadataAccessReq { + // Id of content metadata access + string content_metadata_access_id = 393434598; +} + +message DeleteContentMetadataAccessResponse { + // Successfully deleted. + string result = 1; +} + +message ContentThumbnailReq { + // Either dashboard or look + string type = 3120390; + // ID of the dashboard or look to render + string resource_id = 424027205; + // Whether or not to refresh the rendered image with the latest content + string reload = 334764206; + // A value of png produces a thumbnail in PNG format instead of SVG (default) + string format = 440521963; + // The width of the image if format is supplied + int64 width = 99601414; + // The height of the image if format is supplied + int64 height = 437383491; +} + +message ContentThumbnailResponse { + +} + +message ContentValidationReq { + // Requested fields. + string fields = 453472492; +} + +message ContentValidationResponse { + // Content validation results + ContentValidation result = 1; +} + +message SearchContentViewsReq { + // Match view count + string view_count = 531861914; + // Match Group Id + string group_id = 287220091; + // Match look_id + string look_id = 413287022; + // Match dashboard_id + string dashboard_id = 311679049; + // Match content metadata id + string content_metadata_id = 4111259; + // Match start of week date (format is "YYYY-MM-DD") + string start_of_week_date = 513075243; + // True if only all time view records should be returned + bool all_time = 499982394; + // Match user id + string user_id = 413336787; + // Requested fields + string fields = 453472492; + // Number of results to return. Use with `offset` to manage pagination of results + int64 limit = 110364603; + // Number of results to skip before returning data + int64 offset = 438591449; + // Fields to sort by + string sorts = 109773781; + // Combine given search criteria in a boolean OR expression + bool filter_or = 515276450; +} + +message SearchContentViewsResponse { + // Content View + repeated ContentView result = 1; +} + +message SearchContentViewsStreamResponse { + // Content View + ContentView result = 1; +} + +message VectorThumbnailReq { + // Either dashboard or look + string type = 3120390; + // ID of the dashboard or look to render + string resource_id = 424027205; + // Whether or not to refresh the rendered image with the latest content + string reload = 334764206; +} + +message VectorThumbnailResponse { + // Vector thumbnail + string result = 1; +} + +message AllDashboardsReq { + // Requested fields. + string fields = 453472492; +} + +message AllDashboardsResponse { + // dashboards + repeated DashboardBase result = 1; +} + +message AllDashboardsStreamResponse { + // dashboards + DashboardBase result = 1; +} + +message CreateDashboardReq { + Dashboard body = 3704350; +} + +message CreateDashboardResponse { + // Dashboard + Dashboard result = 1; +} + +message SearchDashboardsReq { + // Match dashboard id. + string id = 3205; + // Match dashboard slug. + string slug = 3184373; + // Match Dashboard title. + string title = 96607896; + // Match Dashboard description. + string description = 375083905; + // Filter on a content favorite id. + string content_favorite_id = 363282886; + // Filter on a particular space. + string folder_id = 271739380; + // Filter on dashboards deleted status. + string deleted = 389460870; + // Filter on dashboards created by a particular user. + string user_id = 413336787; + // Filter on a particular value of view_count + string view_count = 531861914; + // Filter on a content favorite id. + string content_metadata_id = 4111259; + // Exclude items that exist only in personal spaces other than the users + bool curate = 323322900; + // Select dashboards based on when they were last viewed + string last_viewed_at = 510619009; + // Requested fields. + string fields = 453472492; + // DEPRECATED. Use limit and offset instead. Return only page N of paginated results + int64 page = 3110993; + // DEPRECATED. Use limit and offset instead. Return N rows of data per page + int64 per_page = 262811729; + // Number of results to return. (used with offset and takes priority over page and per_page) + int64 limit = 110364603; + // Number of results to skip before returning any. (used with limit and takes priority over page and per_page) + int64 offset = 438591449; + // One or more fields to sort by. Sortable fields: [:title, :user_id, :id, :created_at, :space_id, :folder_id, :description, :view_count, :favorite_count, :slug, :content_favorite_id, :content_metadata_id, :deleted, :deleted_at, :last_viewed_at, :last_accessed_at] + string sorts = 109773781; + // Combine given search criteria in a boolean OR expression + bool filter_or = 515276450; +} + +message SearchDashboardsResponse { + // dashboards + repeated Dashboard result = 1; +} + +message SearchDashboardsStreamResponse { + // dashboards + Dashboard result = 1; +} + +message ImportLookmlDashboardReq { + // Id of LookML dashboard + string lookml_dashboard_id = 351003409; + // Id of space to import the dashboard to + string space_id = 297928564; + Dashboard body = 3704350; + // If true, and this dashboard is localized, export it with the raw keys, not localized. + bool raw_locale = 525398924; +} + +message ImportLookmlDashboardResponse { + // Dashboard + Dashboard result = 1; +} + +message SyncLookmlDashboardReq { + // Id of LookML dashboard, in the form 'model::dashboardname' + string lookml_dashboard_id = 351003409; + Dashboard body = 3704350; + // If true, and this dashboard is localized, export it with the raw keys, not localized. + bool raw_locale = 525398924; +} + +message SyncLookmlDashboardResponse { + // Ids of all the dashboards that were updated by this operation + repeated int64 result = 1; +} + +message SyncLookmlDashboardStreamResponse { + // Ids of all the dashboards that were updated by this operation + int64 result = 1; +} + +message DashboardReq { + // Id of dashboard + string dashboard_id = 311679049; + // Requested fields. + string fields = 453472492; +} + +message DashboardResponse { + // Dashboard + Dashboard result = 1; +} + +message UpdateDashboardReq { + // Id of dashboard + string dashboard_id = 311679049; + Dashboard body = 3704350; +} + +message UpdateDashboardResponse { + // Dashboard + Dashboard result = 1; +} + +message DeleteDashboardReq { + // Id of dashboard + string dashboard_id = 311679049; +} + +message DeleteDashboardResponse { + // Successfully deleted. + string result = 1; +} + +message DashboardAggregateTableLookmlReq { + // Id of dashboard + string dashboard_id = 311679049; +} + +message DashboardAggregateTableLookmlResponse { + // JSON for Aggregate Table LookML + DashboardAggregateTableLookml result = 1; +} + +message DashboardLookmlReq { + // Id of dashboard + string dashboard_id = 311679049; +} + +message DashboardLookmlResponse { + // json of dashboard + DashboardLookml result = 1; +} + +message MoveDashboardReq { + // Dashboard id to move. + string dashboard_id = 311679049; + // Folder id to move to. + string folder_id = 271739380; +} + +message MoveDashboardResponse { + // Dashboard + Dashboard result = 1; +} + +message CreateDashboardFromLookmlReq { + DashboardLookml body = 3704350; +} + +message CreateDashboardFromLookmlResponse { + // DashboardLookML + DashboardLookml result = 1; +} + +message CopyDashboardReq { + // Dashboard id to copy. + string dashboard_id = 311679049; + // Folder id to copy to. + string folder_id = 271739380; +} + +message CopyDashboardResponse { + // Dashboard + Dashboard result = 1; +} + +message SearchDashboardElementsReq { + // Select elements that refer to a given dashboard id + string dashboard_id = 311679049; + // Select elements that refer to a given look id + string look_id = 413287022; + // Match the title of element + string title = 96607896; + // Select soft-deleted dashboard elements + bool deleted = 389460870; + // Requested fields. + string fields = 453472492; + // Combine given search criteria in a boolean OR expression + bool filter_or = 515276450; + // Fields to sort by. Sortable fields: [:look_id, :dashboard_id, :deleted, :title] + string sorts = 109773781; +} + +message SearchDashboardElementsResponse { + // Dashboard elements + repeated DashboardElement result = 1; +} + +message SearchDashboardElementsStreamResponse { + // Dashboard elements + DashboardElement result = 1; +} + +message DashboardElementReq { + // Id of dashboard element + string dashboard_element_id = 426786746; + // Requested fields. + string fields = 453472492; +} + +message DashboardElementResponse { + // DashboardElement + DashboardElement result = 1; +} + +message UpdateDashboardElementReq { + // Id of dashboard element + string dashboard_element_id = 426786746; + DashboardElement body = 3704350; + // Requested fields. + string fields = 453472492; +} + +message UpdateDashboardElementResponse { + // DashboardElement + DashboardElement result = 1; +} + +message DeleteDashboardElementReq { + // Id of dashboard element + string dashboard_element_id = 426786746; +} + +message DeleteDashboardElementResponse { + // Successfully deleted. + string result = 1; +} + +message DashboardDashboardElementsReq { + // Id of dashboard + string dashboard_id = 311679049; + // Requested fields. + string fields = 453472492; +} + +message DashboardDashboardElementsResponse { + // DashboardElement + repeated DashboardElement result = 1; +} + +message DashboardDashboardElementsStreamResponse { + // DashboardElement + DashboardElement result = 1; +} + +message CreateDashboardElementReq { + DashboardElement body = 3704350; + // Requested fields. + string fields = 453472492; + // Apply relevant filters on dashboard to this tile + bool apply_filters = 429426262; +} + +message CreateDashboardElementResponse { + // DashboardElement + DashboardElement result = 1; +} + +message DashboardFilterReq { + // Id of dashboard filters + string dashboard_filter_id = 516692649; + // Requested fields. + string fields = 453472492; +} + +message DashboardFilterResponse { + // Dashboard Filter + DashboardFilter result = 1; +} + +message UpdateDashboardFilterReq { + // Id of dashboard filter + string dashboard_filter_id = 516692649; + DashboardFilter body = 3704350; + // Requested fields. + string fields = 453472492; +} + +message UpdateDashboardFilterResponse { + // Dashboard Filter + DashboardFilter result = 1; +} + +message DeleteDashboardFilterReq { + // Id of dashboard filter + string dashboard_filter_id = 516692649; +} + +message DeleteDashboardFilterResponse { + // Successfully deleted. + string result = 1; +} + +message DashboardDashboardFiltersReq { + // Id of dashboard + string dashboard_id = 311679049; + // Requested fields. + string fields = 453472492; +} + +message DashboardDashboardFiltersResponse { + // Dashboard Filter + repeated DashboardFilter result = 1; +} + +message DashboardDashboardFiltersStreamResponse { + // Dashboard Filter + DashboardFilter result = 1; +} + +message CreateDashboardFilterReq { + CreateDashboardFilter body = 3704350; + // Requested fields + string fields = 453472492; +} + +message CreateDashboardFilterResponse { + // Dashboard Filter + DashboardFilter result = 1; +} + +message DashboardLayoutComponentReq { + // Id of dashboard layout component + string dashboard_layout_component_id = 102016999; + // Requested fields. + string fields = 453472492; +} + +message DashboardLayoutComponentResponse { + // DashboardLayoutComponent + DashboardLayoutComponent result = 1; +} + +message UpdateDashboardLayoutComponentReq { + // Id of dashboard layout component + string dashboard_layout_component_id = 102016999; + DashboardLayoutComponent body = 3704350; + // Requested fields. + string fields = 453472492; +} + +message UpdateDashboardLayoutComponentResponse { + // DashboardLayoutComponent + DashboardLayoutComponent result = 1; +} + +message DashboardLayoutDashboardLayoutComponentsReq { + // Id of dashboard layout component + string dashboard_layout_id = 446158738; + // Requested fields. + string fields = 453472492; +} + +message DashboardLayoutDashboardLayoutComponentsResponse { + // DashboardLayoutComponent + repeated DashboardLayoutComponent result = 1; +} + +message DashboardLayoutDashboardLayoutComponentsStreamResponse { + // DashboardLayoutComponent + DashboardLayoutComponent result = 1; +} + +message DashboardLayoutReq { + // Id of dashboard layouts + string dashboard_layout_id = 446158738; + // Requested fields. + string fields = 453472492; +} + +message DashboardLayoutResponse { + // DashboardLayout + DashboardLayout result = 1; +} + +message UpdateDashboardLayoutReq { + // Id of dashboard layout + string dashboard_layout_id = 446158738; + DashboardLayout body = 3704350; + // Requested fields. + string fields = 453472492; +} + +message UpdateDashboardLayoutResponse { + // DashboardLayout + DashboardLayout result = 1; +} + +message DeleteDashboardLayoutReq { + // Id of dashboard layout + string dashboard_layout_id = 446158738; +} + +message DeleteDashboardLayoutResponse { + // Successfully deleted. + string result = 1; +} + +message DashboardDashboardLayoutsReq { + // Id of dashboard + string dashboard_id = 311679049; + // Requested fields. + string fields = 453472492; +} + +message DashboardDashboardLayoutsResponse { + // DashboardLayout + repeated DashboardLayout result = 1; +} + +message DashboardDashboardLayoutsStreamResponse { + // DashboardLayout + DashboardLayout result = 1; +} + +message CreateDashboardLayoutReq { + DashboardLayout body = 3704350; + // Requested fields. + string fields = 453472492; +} + +message CreateDashboardLayoutResponse { + // DashboardLayout + DashboardLayout result = 1; +} + +message PerformDataActionReq { + DataActionRequest body = 3704350; +} + +message PerformDataActionResponse { + // Data Action Response + DataActionResponse result = 1; +} + +message FetchRemoteDataActionFormReq { + map body = 3704350; +} + +message FetchRemoteDataActionFormResponse { + // Data Action Form + DataActionForm result = 1; +} + +message AllDatagroupsReq { + +} + +message AllDatagroupsResponse { + // Datagroup + repeated Datagroup result = 1; +} + +message AllDatagroupsStreamResponse { + // Datagroup + Datagroup result = 1; +} + +message DatagroupReq { + // ID of datagroup. + string datagroup_id = 448711218; +} + +message DatagroupResponse { + // Datagroup + Datagroup result = 1; +} + +message UpdateDatagroupReq { + // ID of datagroup. + string datagroup_id = 448711218; + Datagroup body = 3704350; +} + +message UpdateDatagroupResponse { + // Datagroup + Datagroup result = 1; +} + +message GraphDerivedTablesForModelReq { + // The name of the Lookml model. + string model = 102848809; + // The format of the graph. Valid values are [dot]. Default is `dot` + string format = 440521963; + // Color denoting the build status of the graph. Grey = not built, green = built, yellow = building, red = error. + string color = 108695523; +} + +message GraphDerivedTablesForModelResponse { + // Derived Table + DependencyGraph result = 1; +} + +message GraphDerivedTablesForViewReq { + // The derived table's view name. + string view = 3645563; + // The models where this derived table is defined. + string models = 449883061; + // The model directory to look in, either `dev` or `production`. + string workspace = 519934677; +} + +message GraphDerivedTablesForViewResponse { + // Graph of the derived table component, represented in the DOT language. + DependencyGraph result = 1; +} + +message StartPdtBuildReq { + // The model of the PDT to start building. + string model_name = 275154383; + // The view name of the PDT to start building. + string view_name = 510594582; + // Force rebuild of required dependent PDTs, even if they are already materialized. + string force_rebuild = 367545223; + // Force involved incremental PDTs to fully re-materialize. + string force_full_incremental = 384123935; + // Workspace in which to materialize selected PDT ('dev' or default 'production'). + string workspace = 519934677; + // The source of this request. + string source = 327120574; +} + +message StartPdtBuildResponse { + // Derived Table + MaterializePDT result = 1; +} + +message CheckPdtBuildReq { + // The materialization id to check status for. + string materialization_id = 347866311; +} + +message CheckPdtBuildResponse { + // Derived Table + MaterializePDT result = 1; +} + +message StopPdtBuildReq { + // The materialization id to stop. + string materialization_id = 347866311; + // The source of this request. + string source = 327120574; +} + +message StopPdtBuildResponse { + // Derived Table + MaterializePDT result = 1; +} + +message SearchFoldersReq { + // Requested fields. + string fields = 453472492; + // Requested page. + int64 page = 3110993; + // Results per page. + int64 per_page = 262811729; + // Number of results to return. (used with offset and takes priority over page and per_page) + int64 limit = 110364603; + // Number of results to skip before returning any. (used with limit and takes priority over page and per_page) + int64 offset = 438591449; + // Fields to sort by. + string sorts = 109773781; + // Match Space title. + string name = 3116757; + // Match Space id + string id = 3205; + // Filter on a children of a particular folder. + string parent_id = 206134256; + // Filter on folder created by a particular user. + string creator_id = 466744818; + // Combine given search criteria in a boolean OR expression + bool filter_or = 515276450; + // Match is shared root + bool is_shared_root = 394142236; +} + +message SearchFoldersResponse { + // folders + repeated Folder result = 1; +} + +message SearchFoldersStreamResponse { + // folders + Folder result = 1; +} + +message FolderReq { + // Id of folder + string folder_id = 271739380; + // Requested fields. + string fields = 453472492; +} + +message FolderResponse { + // Folder + Folder result = 1; +} + +message UpdateFolderReq { + // Id of folder + string folder_id = 271739380; + UpdateFolder body = 3704350; +} + +message UpdateFolderResponse { + // Folder + Folder result = 1; +} + +message DeleteFolderReq { + // Id of folder + string folder_id = 271739380; +} + +message DeleteFolderResponse { + // Successfully deleted. + string result = 1; +} + +message AllFoldersReq { + // Requested fields. + string fields = 453472492; +} + +message AllFoldersResponse { + // Folder + repeated Folder result = 1; +} + +message AllFoldersStreamResponse { + // Folder + Folder result = 1; +} + +message CreateFolderReq { + CreateFolder body = 3704350; +} + +message CreateFolderResponse { + // Folder + Folder result = 1; +} + +message FolderChildrenReq { + // Id of folder + string folder_id = 271739380; + // Requested fields. + string fields = 453472492; + // Requested page. + int64 page = 3110993; + // Results per page. + int64 per_page = 262811729; + // Fields to sort by. + string sorts = 109773781; +} + +message FolderChildrenResponse { + // Folders + repeated Folder result = 1; +} + +message FolderChildrenStreamResponse { + // Folders + Folder result = 1; +} + +message FolderChildrenSearchReq { + // Id of folder + string folder_id = 271739380; + // Requested fields. + string fields = 453472492; + // Fields to sort by. + string sorts = 109773781; + // Match folder name. + string name = 3116757; +} + +message FolderChildrenSearchResponse { + // Folders + repeated Folder result = 1; +} + +message FolderChildrenSearchStreamResponse { + // Folders + Folder result = 1; +} + +message FolderParentReq { + // Id of folder + string folder_id = 271739380; + // Requested fields. + string fields = 453472492; +} + +message FolderParentResponse { + // Folder + Folder result = 1; +} + +message FolderAncestorsReq { + // Id of folder + string folder_id = 271739380; + // Requested fields. + string fields = 453472492; +} + +message FolderAncestorsResponse { + // Folders + repeated Folder result = 1; +} + +message FolderAncestorsStreamResponse { + // Folders + Folder result = 1; +} + +message FolderLooksReq { + // Id of folder + string folder_id = 271739380; + // Requested fields. + string fields = 453472492; +} + +message FolderLooksResponse { + // Looks + repeated LookWithQuery result = 1; +} + +message FolderLooksStreamResponse { + // Looks + LookWithQuery result = 1; +} + +message FolderDashboardsReq { + // Id of folder + string folder_id = 271739380; + // Requested fields. + string fields = 453472492; +} + +message FolderDashboardsResponse { + // Dashboard + repeated Dashboard result = 1; +} + +message FolderDashboardsStreamResponse { + // Dashboard + Dashboard result = 1; +} + +message AllGroupsReq { + // Requested fields. + string fields = 453472492; + // DEPRECATED. Use limit and offset instead. Return only page N of paginated results + int64 page = 3110993; + // DEPRECATED. Use limit and offset instead. Return N rows of data per page + int64 per_page = 262811729; + // Number of results to return. (used with offset and takes priority over page and per_page) + int64 limit = 110364603; + // Number of results to skip before returning any. (used with limit and takes priority over page and per_page) + int64 offset = 438591449; + // Fields to sort by. + string sorts = 109773781; + // Optional of ids to get specific groups. + string ids = 113720; + // Id of content metadata to which groups must have access. + string content_metadata_id = 4111259; + // Select only groups that either can/cannot be given access to content. + bool can_add_to_content_metadata = 454846998; +} + +message AllGroupsResponse { + // Group + repeated Group result = 1; +} + +message AllGroupsStreamResponse { + // Group + Group result = 1; +} + +message CreateGroupReq { + Group body = 3704350; + // Requested fields. + string fields = 453472492; +} + +message CreateGroupResponse { + // Group + Group result = 1; +} + +message SearchGroupsReq { + // Requested fields. + string fields = 453472492; + // Number of results to return (used with `offset`). + int64 limit = 110364603; + // Number of results to skip before returning any (used with `limit`). + int64 offset = 438591449; + // Fields to sort by. + string sorts = 109773781; + // Combine given search criteria in a boolean OR expression + bool filter_or = 515276450; + // Match group id. + string id = 3205; + // Match group name. + string name = 3116757; + // Match group external_group_id. + string external_group_id = 488769304; + // Match group externally_managed. + bool externally_managed = 38023048; + // Match group externally_orphaned. + bool externally_orphaned = 463337449; +} + +message SearchGroupsResponse { + // Group + repeated Group result = 1; +} + +message SearchGroupsStreamResponse { + // Group + Group result = 1; +} + +message SearchGroupsWithRolesReq { + // Requested fields. + string fields = 453472492; + // Number of results to return (used with `offset`). + int64 limit = 110364603; + // Number of results to skip before returning any (used with `limit`). + int64 offset = 438591449; + // Fields to sort by. + string sorts = 109773781; + // Combine given search criteria in a boolean OR expression + bool filter_or = 515276450; + // Match group id. + string id = 3205; + // Match group name. + string name = 3116757; + // Match group external_group_id. + string external_group_id = 488769304; + // Match group externally_managed. + bool externally_managed = 38023048; + // Match group externally_orphaned. + bool externally_orphaned = 463337449; +} + +message SearchGroupsWithRolesResponse { + // Group + repeated GroupSearch result = 1; +} + +message SearchGroupsWithRolesStreamResponse { + // Group + GroupSearch result = 1; +} + +message SearchGroupsWithHierarchyReq { + // Requested fields. + string fields = 453472492; + // Number of results to return (used with `offset`). + int64 limit = 110364603; + // Number of results to skip before returning any (used with `limit`). + int64 offset = 438591449; + // Fields to sort by. + string sorts = 109773781; + // Combine given search criteria in a boolean OR expression + bool filter_or = 515276450; + // Match group id. + string id = 3205; + // Match group name. + string name = 3116757; + // Match group external_group_id. + string external_group_id = 488769304; + // Match group externally_managed. + bool externally_managed = 38023048; + // Match group externally_orphaned. + bool externally_orphaned = 463337449; +} + +message SearchGroupsWithHierarchyResponse { + // Group + repeated GroupHierarchy result = 1; +} + +message SearchGroupsWithHierarchyStreamResponse { + // Group + GroupHierarchy result = 1; +} + +message GroupReq { + // Id of group + string group_id = 287220091; + // Requested fields. + string fields = 453472492; +} + +message GroupResponse { + // Group + Group result = 1; +} + +message UpdateGroupReq { + // Id of group + string group_id = 287220091; + Group body = 3704350; + // Requested fields. + string fields = 453472492; +} + +message UpdateGroupResponse { + // Group + Group result = 1; +} + +message DeleteGroupReq { + // Id of group + string group_id = 287220091; +} + +message DeleteGroupResponse { + // Successfully deleted. + string result = 1; +} + +message AllGroupGroupsReq { + // Id of group + string group_id = 287220091; + // Requested fields. + string fields = 453472492; +} + +message AllGroupGroupsResponse { + // All groups in group. + repeated Group result = 1; +} + +message AllGroupGroupsStreamResponse { + // All groups in group. + Group result = 1; +} + +message AddGroupGroupReq { + // Id of group + string group_id = 287220091; + GroupIdForGroupInclusion body = 3704350; +} + +message AddGroupGroupResponse { + // Added group. + Group result = 1; +} + +message AllGroupUsersReq { + // Id of group + string group_id = 287220091; + // Requested fields. + string fields = 453472492; + // DEPRECATED. Use limit and offset instead. Return only page N of paginated results + int64 page = 3110993; + // DEPRECATED. Use limit and offset instead. Return N rows of data per page + int64 per_page = 262811729; + // Number of results to return. (used with offset and takes priority over page and per_page) + int64 limit = 110364603; + // Number of results to skip before returning any. (used with limit and takes priority over page and per_page) + int64 offset = 438591449; + // Fields to sort by. + string sorts = 109773781; +} + +message AllGroupUsersResponse { + // All users in group. + repeated User result = 1; +} + +message AllGroupUsersStreamResponse { + // All users in group. + User result = 1; +} + +message AddGroupUserReq { + // Id of group + string group_id = 287220091; + GroupIdForGroupUserInclusion body = 3704350; +} + +message AddGroupUserResponse { + // Added user. + User result = 1; +} + +message DeleteGroupUserReq { + // Id of group + string group_id = 287220091; + // Id of user to remove from group + string user_id = 413336787; +} + +message DeleteGroupUserResponse { + // User successfully removed from group +} + +message DeleteGroupFromGroupReq { + // Id of group + string group_id = 287220091; + // Id of group to delete + string deleting_group_id = 426162644; +} + +message DeleteGroupFromGroupResponse { + // Group successfully deleted +} + +message UpdateUserAttributeGroupValueReq { + // Id of group + string group_id = 287220091; + // Id of user attribute + string user_attribute_id = 18993938; + UserAttributeGroupValue body = 3704350; +} + +message UpdateUserAttributeGroupValueResponse { + // Group value object. + UserAttributeGroupValue result = 1; +} + +message DeleteUserAttributeGroupValueReq { + // Id of group + string group_id = 287220091; + // Id of user attribute + string user_attribute_id = 18993938; +} + +message DeleteUserAttributeGroupValueResponse { + // Value successfully unset +} + +message AllPrimaryHomepageSectionsReq { + // Requested fields. + string fields = 453472492; +} + +message AllPrimaryHomepageSectionsResponse { + // Primary homepage section + repeated HomepageSection result = 1; +} + +message AllPrimaryHomepageSectionsStreamResponse { + // Primary homepage section + HomepageSection result = 1; +} + +message AllIntegrationHubsReq { + // Requested fields. + string fields = 453472492; +} + +message AllIntegrationHubsResponse { + // Integration Hub + repeated IntegrationHub result = 1; +} + +message AllIntegrationHubsStreamResponse { + // Integration Hub + IntegrationHub result = 1; +} + +message CreateIntegrationHubReq { + IntegrationHub body = 3704350; + // Requested fields. + string fields = 453472492; +} + +message CreateIntegrationHubResponse { + // Integration Hub + IntegrationHub result = 1; +} + +message IntegrationHubReq { + // Id of integration_hub + string integration_hub_id = 469183532; + // Requested fields. + string fields = 453472492; +} + +message IntegrationHubResponse { + // Integration Hub + IntegrationHub result = 1; +} + +message UpdateIntegrationHubReq { + // Id of integration_hub + string integration_hub_id = 469183532; + IntegrationHub body = 3704350; + // Requested fields. + string fields = 453472492; +} + +message UpdateIntegrationHubResponse { + // Integration Hub + IntegrationHub result = 1; +} + +message DeleteIntegrationHubReq { + // Id of integration_hub + string integration_hub_id = 469183532; +} + +message DeleteIntegrationHubResponse { + // Successfully deleted. + string result = 1; +} + +message AcceptIntegrationHubLegalAgreementReq { + // Id of integration_hub + string integration_hub_id = 469183532; +} + +message AcceptIntegrationHubLegalAgreementResponse { + // Integration hub + IntegrationHub result = 1; +} + +message AllIntegrationsReq { + // Requested fields. + string fields = 453472492; + // Filter to a specific provider + string integration_hub_id = 469183532; +} + +message AllIntegrationsResponse { + // Integration + repeated Integration result = 1; +} + +message AllIntegrationsStreamResponse { + // Integration + Integration result = 1; +} + +message IntegrationReq { + // Id of integration + string integration_id = 515316921; + // Requested fields. + string fields = 453472492; +} + +message IntegrationResponse { + // Integration + Integration result = 1; +} + +message UpdateIntegrationReq { + // Id of integration + string integration_id = 515316921; + Integration body = 3704350; + // Requested fields. + string fields = 453472492; +} + +message UpdateIntegrationResponse { + // Integration + Integration result = 1; +} + +message FetchIntegrationFormReq { + // Id of integration + string integration_id = 515316921; + map body = 3704350; +} + +message FetchIntegrationFormResponse { + // Data Action Form + DataActionForm result = 1; +} + +message TestIntegrationReq { + // Id of integration + string integration_id = 515316921; +} + +message TestIntegrationResponse { + // Test Result + IntegrationTestResult result = 1; +} + +message AllLooksReq { + // Requested fields. + string fields = 453472492; +} + +message AllLooksResponse { + // Look + repeated Look result = 1; +} + +message AllLooksStreamResponse { + // Look + Look result = 1; +} + +message CreateLookReq { + LookWithQuery body = 3704350; + // Requested fields. + string fields = 453472492; +} + +message CreateLookResponse { + // Look + LookWithQuery result = 1; +} + +message SearchLooksReq { + // Match look id. + string id = 3205; + // Match Look title. + string title = 96607896; + // Match Look description. + string description = 375083905; + // Select looks with a particular content favorite id + string content_favorite_id = 363282886; + // Select looks in a particular folder. + string folder_id = 271739380; + // Select looks created by a particular user. + string user_id = 413336787; + // Select looks with particular view_count value + string view_count = 531861914; + // Select soft-deleted looks + bool deleted = 389460870; + // Select looks that reference a particular query by query_id + string query_id = 279007282; + // Exclude items that exist only in personal spaces other than the users + bool curate = 323322900; + // Select looks based on when they were last viewed + string last_viewed_at = 510619009; + // Requested fields. + string fields = 453472492; + // DEPRECATED. Use limit and offset instead. Return only page N of paginated results + int64 page = 3110993; + // DEPRECATED. Use limit and offset instead. Return N rows of data per page + int64 per_page = 262811729; + // Number of results to return. (used with offset and takes priority over page and per_page) + int64 limit = 110364603; + // Number of results to skip before returning any. (used with limit and takes priority over page and per_page) + int64 offset = 438591449; + // One or more fields to sort results by. Sortable fields: [:title, :user_id, :id, :created_at, :space_id, :folder_id, :description, :updated_at, :last_updater_id, :view_count, :favorite_count, :content_favorite_id, :deleted, :deleted_at, :last_viewed_at, :last_accessed_at, :query_id] + string sorts = 109773781; + // Combine given search criteria in a boolean OR expression + bool filter_or = 515276450; +} + +message SearchLooksResponse { + // looks + repeated Look result = 1; +} + +message SearchLooksStreamResponse { + // looks + Look result = 1; +} + +message LookReq { + // Id of look + string look_id = 413287022; + // Requested fields. + string fields = 453472492; +} + +message LookResponse { + // Look + LookWithQuery result = 1; +} + +message UpdateLookReq { + // Id of look + string look_id = 413287022; + LookWithQuery body = 3704350; + // Requested fields. + string fields = 453472492; +} + +message UpdateLookResponse { + // Look + LookWithQuery result = 1; +} + +message DeleteLookReq { + // Id of look + string look_id = 413287022; +} + +message DeleteLookResponse { + // Successfully deleted. + string result = 1; +} + +message RunLookReq { + // Id of look + string look_id = 413287022; + // Format of result + string result_format = 296518982; + // Row limit (may override the limit in the saved query). + int64 limit = 110364603; + // Apply model-specified formatting to each result. + bool apply_formatting = 423291115; + // Apply visualization options to results. + bool apply_vis = 273430419; + // Get results from cache if available. + bool cache = 96472130; + // Render width for image formats. + int64 image_width = 215473054; + // Render height for image formats. + int64 image_height = 438315466; + // Generate drill links (only applicable to 'json_detail' format. + bool generate_drill_links = 524243882; + // Force use of production models even if the user is in development mode. + bool force_production = 426087380; + // Retrieve any results from cache even if the results have expired. + bool cache_only = 520313387; + // Prefix to use for drill links (url encoded). + string path_prefix = 400540043; + // Rebuild PDTS used in query. + bool rebuild_pdts = 367772772; + // Perform table calculations on query results + bool server_table_calcs = 437532812; +} + +message RunLookResponse { + +} + +message CopyLookReq { + // Look id to copy. + string look_id = 413287022; + // Folder id to copy to. + string folder_id = 271739380; +} + +message CopyLookResponse { + // Look + LookWithQuery result = 1; +} + +message MoveLookReq { + // Look id to move. + string look_id = 413287022; + // Folder id to move to. + string folder_id = 271739380; +} + +message MoveLookResponse { + // Look + LookWithQuery result = 1; +} + +message AllLookmlModelsReq { + // Requested fields. + string fields = 453472492; + // Number of results to return. (can be used with offset) + int64 limit = 110364603; + // Number of results to skip before returning any. (Defaults to 0 if not set when limit is used) + int64 offset = 438591449; +} + +message AllLookmlModelsResponse { + // LookML Model + repeated LookmlModel result = 1; +} + +message AllLookmlModelsStreamResponse { + // LookML Model + LookmlModel result = 1; +} + +message CreateLookmlModelReq { + LookmlModel body = 3704350; +} + +message CreateLookmlModelResponse { + // LookML Model + LookmlModel result = 1; +} + +message LookmlModelReq { + // Name of lookml model. + string lookml_model_name = 465378271; + // Requested fields. + string fields = 453472492; +} + +message LookmlModelResponse { + // LookML Model + LookmlModel result = 1; +} + +message UpdateLookmlModelReq { + // Name of lookml model. + string lookml_model_name = 465378271; + LookmlModel body = 3704350; +} + +message UpdateLookmlModelResponse { + // LookML Model + LookmlModel result = 1; +} + +message DeleteLookmlModelReq { + // Name of lookml model. + string lookml_model_name = 465378271; +} + +message DeleteLookmlModelResponse { + // Successfully deleted. + string result = 1; +} + +message LookmlModelExploreReq { + // Name of lookml model. + string lookml_model_name = 465378271; + // Name of explore. + string explore_name = 403807127; + // Requested fields. + string fields = 453472492; +} + +message LookmlModelExploreResponse { + // LookML Model Explore + LookmlModelExplore result = 1; +} + +message ModelFieldnameSuggestionsReq { + // Name of model + string model_name = 275154383; + // Name of view + string view_name = 510594582; + // Name of field to use for suggestions + string field_name = 273359668; + // Search term pattern (evaluated as as `%term%`) + string term = 3360020; + // Suggestion filters with field name keys and comparison expressions + google.protobuf.Any filters = 487674337; +} + +message ModelFieldnameSuggestionsResponse { + // Model view field suggestions + ModelFieldSuggestions result = 1; +} + +message GetModelReq { + // Name of model + string model_name = 275154383; +} + +message GetModelResponse { + // A Model + Model result = 1; +} + +message ConnectionDatabasesReq { + // Name of connection + string connection_name = 292836428; +} + +message ConnectionDatabasesResponse { + // Database names + repeated string result = 1; +} + +message ConnectionDatabasesStreamResponse { + // Database names + string result = 1; +} + +message ConnectionFeaturesReq { + // Name of connection + string connection_name = 292836428; + // Requested fields. + string fields = 453472492; +} + +message ConnectionFeaturesResponse { + // Connection features + ConnectionFeatures result = 1; +} + +message ConnectionSchemasReq { + // Name of connection + string connection_name = 292836428; + // For dialects that support multiple databases, optionally identify which to use + string database = 446107481; + // True to use fetch from cache, false to load fresh + bool cache = 96472130; + // Requested fields. + string fields = 453472492; +} + +message ConnectionSchemasResponse { + // Schemas for connection + repeated Schema result = 1; +} + +message ConnectionSchemasStreamResponse { + // Schemas for connection + Schema result = 1; +} + +message ConnectionTablesReq { + // Name of connection + string connection_name = 292836428; + // Optional. Name of database to use for the query, only if applicable + string database = 446107481; + // Optional. Return only tables for this schema + string schema_name = 274052203; + // True to fetch from cache, false to load fresh + bool cache = 96472130; + // Requested fields. + string fields = 453472492; + // Optional. Return tables with names that contain this value + string table_filter = 533151572; + // Optional. Return tables up to the table_limit + int64 table_limit = 399479586; +} + +message ConnectionTablesResponse { + // Schemas and tables for connection + repeated SchemaTables result = 1; +} + +message ConnectionTablesStreamResponse { + // Schemas and tables for connection + SchemaTables result = 1; +} + +message ConnectionColumnsReq { + // Name of connection + string connection_name = 292836428; + // For dialects that support multiple databases, optionally identify which to use + string database = 446107481; + // Name of schema to use. + string schema_name = 274052203; + // True to fetch from cache, false to load fresh + bool cache = 96472130; + // limits the tables per schema returned + int64 table_limit = 399479586; + // only fetch columns for a given (comma-separated) list of tables + string table_names = 301691499; + // Requested fields. + string fields = 453472492; +} + +message ConnectionColumnsResponse { + // Columns schema for connection + repeated SchemaColumns result = 1; +} + +message ConnectionColumnsStreamResponse { + // Columns schema for connection + SchemaColumns result = 1; +} + +message ConnectionSearchColumnsReq { + // Name of connection + string connection_name = 292836428; + // Column name to find + string column_name = 175444588; + // Requested fields. + string fields = 453472492; +} + +message ConnectionSearchColumnsResponse { + // Column names matching search pattern + repeated ColumnSearch result = 1; +} + +message ConnectionSearchColumnsStreamResponse { + // Column names matching search pattern + ColumnSearch result = 1; +} + +message ConnectionCostEstimateReq { + // Name of connection + string connection_name = 292836428; + CreateCostEstimate body = 3704350; + // Requested fields. + string fields = 453472492; +} + +message ConnectionCostEstimateResponse { + // Connection cost estimates + CostEstimate result = 1; +} + +message LockAllReq { + // Id of project + string project_id = 471156823; + // Requested fields + string fields = 453472492; +} + +message LockAllResponse { + +} + +message AllGitBranchesReq { + // Project Id + string project_id = 471156823; +} + +message AllGitBranchesResponse { + // Git Branch + repeated GitBranch result = 1; +} + +message AllGitBranchesStreamResponse { + // Git Branch + GitBranch result = 1; +} + +message GitBranchReq { + // Project Id + string project_id = 471156823; +} + +message GitBranchResponse { + // Git Branch + GitBranch result = 1; +} + +message UpdateGitBranchReq { + // Project Id + string project_id = 471156823; + GitBranch body = 3704350; +} + +message UpdateGitBranchResponse { + // Git Branch + GitBranch result = 1; +} + +message CreateGitBranchReq { + // Project Id + string project_id = 471156823; + GitBranch body = 3704350; +} + +message CreateGitBranchResponse { + // Git Branch + GitBranch result = 1; +} + +message FindGitBranchReq { + // Project Id + string project_id = 471156823; + // Branch Name + string branch_name = 356673720; +} + +message FindGitBranchResponse { + // Git Branch + GitBranch result = 1; +} + +message DeleteGitBranchReq { + // Project Id + string project_id = 471156823; + // Branch Name + string branch_name = 356673720; +} + +message DeleteGitBranchResponse { + // Successfully deleted. + string result = 1; +} + +message DeployRefToProductionReq { + // Id of project + string project_id = 471156823; + // Branch to deploy to production + string branch = 305683288; + // Ref to deploy to production + string ref = 101267; +} + +message DeployRefToProductionResponse { + +} + +message DeployToProductionReq { + // Id of project + string project_id = 471156823; +} + +message DeployToProductionResponse { + +} + +message ResetProjectToProductionReq { + // Id of project + string project_id = 471156823; +} + +message ResetProjectToProductionResponse { + +} + +message ResetProjectToRemoteReq { + // Id of project + string project_id = 471156823; +} + +message ResetProjectToRemoteResponse { + +} + +message AllProjectsReq { + // Requested fields + string fields = 453472492; +} + +message AllProjectsResponse { + // Project + repeated Project result = 1; +} + +message AllProjectsStreamResponse { + // Project + Project result = 1; +} + +message CreateProjectReq { + Project body = 3704350; +} + +message CreateProjectResponse { + // Project + Project result = 1; +} + +message ProjectReq { + // Project Id + string project_id = 471156823; + // Requested fields + string fields = 453472492; +} + +message ProjectResponse { + // Project + Project result = 1; +} + +message UpdateProjectReq { + // Project Id + string project_id = 471156823; + Project body = 3704350; + // Requested fields + string fields = 453472492; +} + +message UpdateProjectResponse { + // Project + Project result = 1; +} + +message ManifestReq { + // Project Id + string project_id = 471156823; +} + +message ManifestResponse { + // Manifest + Manifest result = 1; +} + +message GitDeployKeyReq { + // Project Id + string project_id = 471156823; +} + +message GitDeployKeyResponse { + // The text of the public key portion of the deploy_key + string result = 1; +} + +message CreateGitDeployKeyReq { + // Project Id + string project_id = 471156823; +} + +message CreateGitDeployKeyResponse { + // Project + string result = 1; +} + +message ProjectValidationResultsReq { + // Project Id + string project_id = 471156823; + // Requested fields + string fields = 453472492; +} + +message ProjectValidationResultsResponse { + +} + +message ValidateProjectReq { + // Project Id + string project_id = 471156823; + // Requested fields + string fields = 453472492; +} + +message ValidateProjectResponse { + // Project validation results + ProjectValidation result = 1; +} + +message ProjectWorkspaceReq { + // Project Id + string project_id = 471156823; + // Requested fields + string fields = 453472492; +} + +message ProjectWorkspaceResponse { + // Project Workspace + ProjectWorkspace result = 1; +} + +message AllProjectFilesReq { + // Project Id + string project_id = 471156823; + // Requested fields + string fields = 453472492; +} + +message AllProjectFilesResponse { + // Project File + repeated ProjectFile result = 1; +} + +message AllProjectFilesStreamResponse { + // Project File + ProjectFile result = 1; +} + +message ProjectFileReq { + // Project Id + string project_id = 471156823; + // File Id + string file_id = 413241567; + // Requested fields + string fields = 453472492; +} + +message ProjectFileResponse { + // Project File + ProjectFile result = 1; +} + +message AllGitConnectionTestsReq { + // Project Id + string project_id = 471156823; + // (Optional: leave blank for root project) The remote url for remote dependency to test. + string remote_url = 474030390; +} + +message AllGitConnectionTestsResponse { + // Git Connection Test + repeated GitConnectionTest result = 1; +} + +message AllGitConnectionTestsStreamResponse { + // Git Connection Test + GitConnectionTest result = 1; +} + +message RunGitConnectionTestReq { + // Project Id + string project_id = 471156823; + // Test Id + string test_id = 413354938; + // (Optional: leave blank for root project) The remote url for remote dependency to test. + string remote_url = 474030390; + // (Optional: leave blank for dev credentials) Whether to use git production credentials. + string use_production = 402908004; +} + +message RunGitConnectionTestResponse { + // Git Connection Test Result + GitConnectionTestResult result = 1; +} + +message AllLookmlTestsReq { + // Project Id + string project_id = 471156823; + // File Id + string file_id = 413241567; +} + +message AllLookmlTestsResponse { + // LookML Test + repeated LookmlTest result = 1; +} + +message AllLookmlTestsStreamResponse { + // LookML Test + LookmlTest result = 1; +} + +message RunLookmlTestReq { + // Project Id + string project_id = 471156823; + // File Name + string file_id = 413241567; + // Test Name + string test = 3569518; + // Model Name + string model = 102848809; +} + +message RunLookmlTestResponse { + // LookML Test Results + repeated LookmlTestResult result = 1; +} + +message RunLookmlTestStreamResponse { + // LookML Test Results + LookmlTestResult result = 1; +} + +message TagRefReq { + // Project Id + string project_id = 471156823; + Project body = 3704350; + // (Optional): Commit Sha to Tag + string commit_sha = 484107812; + // Tag Name + string tag_name = 382692664; + // (Optional): Tag Message + string tag_message = 359908744; +} + +message TagRefResponse { + +} + +message UpdateRepositoryCredentialReq { + // Root Project Id + string root_project_id = 487670047; + // Credential Id + string credential_id = 318409473; + RepositoryCredential body = 3704350; +} + +message UpdateRepositoryCredentialResponse { + // Repository Credential + RepositoryCredential result = 1; +} + +message DeleteRepositoryCredentialReq { + // Root Project Id + string root_project_id = 487670047; + // Credential Id + string credential_id = 318409473; +} + +message DeleteRepositoryCredentialResponse { + // Successfully deleted. + string result = 1; +} + +message GetAllRepositoryCredentialsReq { + // Root Project Id + string root_project_id = 487670047; +} + +message GetAllRepositoryCredentialsResponse { + // Repository Credential + repeated RepositoryCredential result = 1; +} + +message GetAllRepositoryCredentialsStreamResponse { + // Repository Credential + RepositoryCredential result = 1; +} + +message CreateQueryTaskReq { + CreateQueryTask body = 3704350; + // Row limit (may override the limit in the saved query). + int64 limit = 110364603; + // Apply model-specified formatting to each result. + bool apply_formatting = 423291115; + // Apply visualization options to results. + bool apply_vis = 273430419; + // Get results from cache if available. + bool cache = 96472130; + // Render width for image formats. + int64 image_width = 215473054; + // Render height for image formats. + int64 image_height = 438315466; + // Generate drill links (only applicable to 'json_detail' format. + bool generate_drill_links = 524243882; + // Force use of production models even if the user is in development mode. + bool force_production = 426087380; + // Retrieve any results from cache even if the results have expired. + bool cache_only = 520313387; + // Prefix to use for drill links (url encoded). + string path_prefix = 400540043; + // Rebuild PDTS used in query. + bool rebuild_pdts = 367772772; + // Perform table calculations on query results + bool server_table_calcs = 437532812; + // Requested fields + string fields = 453472492; +} + +message CreateQueryTaskResponse { + // query_task + QueryTask result = 1; +} + +message QueryTaskMultiResultsReq { + // List of Query Task IDs + string query_task_ids = 278476749; +} + +message QueryTaskMultiResultsResponse { + // Multiple query results + map result = 1; +} + +message QueryTaskReq { + // ID of the Query Task + string query_task_id = 297690007; + // Requested fields. + string fields = 453472492; +} + +message QueryTaskResponse { + // query_task + QueryTask result = 1; +} + +message QueryTaskResultsReq { + // ID of the Query Task + string query_task_id = 297690007; +} + +message QueryTaskResultsResponse { + +} + +message QueryReq { + // Id of query + string query_id = 279007282; + // Requested fields. + string fields = 453472492; +} + +message QueryResponse { + // Query + Query result = 1; +} + +message QueryForSlugReq { + // Slug of query + string slug = 3184373; + // Requested fields. + string fields = 453472492; +} + +message QueryForSlugResponse { + // Query + Query result = 1; +} + +message CreateQueryReq { + Query body = 3704350; + // Requested fields. + string fields = 453472492; +} + +message CreateQueryResponse { + // Query + Query result = 1; +} + +message RunQueryReq { + // Id of query + string query_id = 279007282; + // Format of result + string result_format = 296518982; + // Row limit (may override the limit in the saved query). + int64 limit = 110364603; + // Apply model-specified formatting to each result. + bool apply_formatting = 423291115; + // Apply visualization options to results. + bool apply_vis = 273430419; + // Get results from cache if available. + bool cache = 96472130; + // Render width for image formats. + int64 image_width = 215473054; + // Render height for image formats. + int64 image_height = 438315466; + // Generate drill links (only applicable to 'json_detail' format. + bool generate_drill_links = 524243882; + // Force use of production models even if the user is in development mode. + bool force_production = 426087380; + // Retrieve any results from cache even if the results have expired. + bool cache_only = 520313387; + // Prefix to use for drill links (url encoded). + string path_prefix = 400540043; + // Rebuild PDTS used in query. + bool rebuild_pdts = 367772772; + // Perform table calculations on query results + bool server_table_calcs = 437532812; + // Specifies the source of this call. + string source = 327120574; +} + +message RunQueryResponse { + +} + +message RunInlineQueryReq { + // Format of result + string result_format = 296518982; + Query body = 3704350; + // Row limit (may override the limit in the saved query). + int64 limit = 110364603; + // Apply model-specified formatting to each result. + bool apply_formatting = 423291115; + // Apply visualization options to results. + bool apply_vis = 273430419; + // Get results from cache if available. + bool cache = 96472130; + // Render width for image formats. + int64 image_width = 215473054; + // Render height for image formats. + int64 image_height = 438315466; + // Generate drill links (only applicable to 'json_detail' format. + bool generate_drill_links = 524243882; + // Force use of production models even if the user is in development mode. + bool force_production = 426087380; + // Retrieve any results from cache even if the results have expired. + bool cache_only = 520313387; + // Prefix to use for drill links (url encoded). + string path_prefix = 400540043; + // Rebuild PDTS used in query. + bool rebuild_pdts = 367772772; + // Perform table calculations on query results + bool server_table_calcs = 437532812; +} + +message RunInlineQueryResponse { + +} + +message RunUrlEncodedQueryReq { + // Model name + string model_name = 275154383; + // View name + string view_name = 510594582; + // Format of result + string result_format = 296518982; +} + +message RunUrlEncodedQueryResponse { + +} + +message MergeQueryReq { + // Merge Query Id + string merge_query_id = 209670407; + // Requested fields + string fields = 453472492; +} + +message MergeQueryResponse { + // Merge Query + MergeQuery result = 1; +} + +message CreateMergeQueryReq { + MergeQuery body = 3704350; + // Requested fields + string fields = 453472492; +} + +message CreateMergeQueryResponse { + // Merge Query + MergeQuery result = 1; +} + +message AllRunningQueriesReq { + +} + +message AllRunningQueriesResponse { + // Running Queries. + repeated RunningQueries result = 1; +} + +message AllRunningQueriesStreamResponse { + // Running Queries. + RunningQueries result = 1; +} + +message KillQueryReq { + // Query task id. + string query_task_id = 297690007; +} + +message KillQueryResponse { + // Query successfully killed. + string result = 1; +} + +message SqlQueryReq { + // slug of query + string slug = 3184373; +} + +message SqlQueryResponse { + // SQL Runner Query + SqlQuery result = 1; +} + +message CreateSqlQueryReq { + SqlQueryCreate body = 3704350; +} + +message CreateSqlQueryResponse { + // SQL Runner Query + SqlQuery result = 1; +} + +message RunSqlQueryReq { + // slug of query + string slug = 3184373; + // Format of result, options are: ["inline_json", "json", "json_detail", "json_fe", "csv", "html", "md", "txt", "xlsx", "gsxml", "json_label"] + string result_format = 296518982; + // Defaults to false. If set to true, the HTTP response will have content-disposition and other headers set to make the HTTP response behave as a downloadable attachment instead of as inline content. + string download = 414214110; +} + +message RunSqlQueryResponse { + +} + +message CreateLookRenderTaskReq { + // Id of look to render + string look_id = 413287022; + // Output type: png, or jpg + string result_format = 296518982; + // Output width in pixels + int64 width = 99601414; + // Output height in pixels + int64 height = 437383491; + // Requested fields. + string fields = 453472492; +} + +message CreateLookRenderTaskResponse { + // Render Task + RenderTask result = 1; +} + +message CreateQueryRenderTaskReq { + // Id of the query to render + string query_id = 279007282; + // Output type: png or jpg + string result_format = 296518982; + // Output width in pixels + int64 width = 99601414; + // Output height in pixels + int64 height = 437383491; + // Requested fields. + string fields = 453472492; +} + +message CreateQueryRenderTaskResponse { + // Render Task + RenderTask result = 1; +} + +message CreateDashboardRenderTaskReq { + // Id of dashboard to render. The ID can be a LookML dashboard also. + string dashboard_id = 311679049; + // Output type: pdf, png, or jpg + string result_format = 296518982; + CreateDashboardRenderTask body = 3704350; + // Output width in pixels + int64 width = 99601414; + // Output height in pixels + int64 height = 437383491; + // Requested fields. + string fields = 453472492; + // Paper size for pdf. Value can be one of: ["letter","legal","tabloid","a0","a1","a2","a3","a4","a5"] + string pdf_paper_size = 317503911; + // Whether to render pdf in landscape paper orientation + bool pdf_landscape = 272762770; + // Whether or not to expand table vis to full length + bool long_tables = 450392146; +} + +message CreateDashboardRenderTaskResponse { + // Render Task + RenderTask result = 1; +} + +message RenderTaskReq { + // Id of render task + string render_task_id = 488397835; + // Requested fields. + string fields = 453472492; +} + +message RenderTaskResponse { + // Render Task + RenderTask result = 1; +} + +message RenderTaskResultsReq { + // Id of render task + string render_task_id = 488397835; +} + +message RenderTaskResultsResponse { + +} + +message CreateDashboardElementRenderTaskReq { + // Id of dashboard element to render: UDD dashboard element would be numeric and LookML dashboard element would be model_name::dashboard_title::lookml_link_id + string dashboard_element_id = 426786746; + // Output type: png or jpg + string result_format = 296518982; + // Output width in pixels + int64 width = 99601414; + // Output height in pixels + int64 height = 437383491; + // Requested fields. + string fields = 453472492; +} + +message CreateDashboardElementRenderTaskResponse { + // Render Task + RenderTask result = 1; +} + +message SearchModelSetsReq { + // Requested fields. + string fields = 453472492; + // Number of results to return (used with `offset`). + int64 limit = 110364603; + // Number of results to skip before returning any (used with `limit`). + int64 offset = 438591449; + // Fields to sort by. + string sorts = 109773781; + // Match model set id. + string id = 3205; + // Match model set name. + string name = 3116757; + // Match model sets by all_access status. + bool all_access = 499364607; + // Match model sets by built_in status. + bool built_in = 35565638; + // Combine given search criteria in a boolean OR expression. + bool filter_or = 515276450; +} + +message SearchModelSetsResponse { + // Model Set + repeated ModelSet result = 1; +} + +message SearchModelSetsStreamResponse { + // Model Set + ModelSet result = 1; +} + +message ModelSetReq { + // Id of model set + string model_set_id = 339578852; + // Requested fields. + string fields = 453472492; +} + +message ModelSetResponse { + // Specified model set. + ModelSet result = 1; +} + +message UpdateModelSetReq { + // id of model set + string model_set_id = 339578852; + ModelSet body = 3704350; +} + +message UpdateModelSetResponse { + // New state for specified model set. + ModelSet result = 1; +} + +message DeleteModelSetReq { + // id of model set + string model_set_id = 339578852; +} + +message DeleteModelSetResponse { + // Model set successfully deleted. + string result = 1; +} + +message AllModelSetsReq { + // Requested fields. + string fields = 453472492; +} + +message AllModelSetsResponse { + // All model sets. + repeated ModelSet result = 1; +} + +message AllModelSetsStreamResponse { + // All model sets. + ModelSet result = 1; +} + +message CreateModelSetReq { + ModelSet body = 3704350; +} + +message CreateModelSetResponse { + // Newly created ModelSet + ModelSet result = 1; +} + +message AllPermissionsReq { + +} + +message AllPermissionsResponse { + // Permission + repeated Permission result = 1; +} + +message AllPermissionsStreamResponse { + // Permission + Permission result = 1; +} + +message SearchPermissionSetsReq { + // Requested fields. + string fields = 453472492; + // Number of results to return (used with `offset`). + int64 limit = 110364603; + // Number of results to skip before returning any (used with `limit`). + int64 offset = 438591449; + // Fields to sort by. + string sorts = 109773781; + // Match permission set id. + string id = 3205; + // Match permission set name. + string name = 3116757; + // Match permission sets by all_access status. + bool all_access = 499364607; + // Match permission sets by built_in status. + bool built_in = 35565638; + // Combine given search criteria in a boolean OR expression. + bool filter_or = 515276450; +} + +message SearchPermissionSetsResponse { + // Permission Set + repeated PermissionSet result = 1; +} + +message SearchPermissionSetsStreamResponse { + // Permission Set + PermissionSet result = 1; +} + +message PermissionSetReq { + // Id of permission set + string permission_set_id = 239948280; + // Requested fields. + string fields = 453472492; +} + +message PermissionSetResponse { + // Permission Set + PermissionSet result = 1; +} + +message UpdatePermissionSetReq { + // Id of permission set + string permission_set_id = 239948280; + PermissionSet body = 3704350; +} + +message UpdatePermissionSetResponse { + // Permission Set + PermissionSet result = 1; +} + +message DeletePermissionSetReq { + // Id of permission set + string permission_set_id = 239948280; +} + +message DeletePermissionSetResponse { + // Successfully deleted. + string result = 1; +} + +message AllPermissionSetsReq { + // Requested fields. + string fields = 453472492; +} + +message AllPermissionSetsResponse { + // Permission Set + repeated PermissionSet result = 1; +} + +message AllPermissionSetsStreamResponse { + // Permission Set + PermissionSet result = 1; +} + +message CreatePermissionSetReq { + PermissionSet body = 3704350; +} + +message CreatePermissionSetResponse { + // Permission Set + PermissionSet result = 1; +} + +message AllRolesReq { + // Requested fields. + string fields = 453472492; + // Optional list of ids to get specific roles. + string ids = 113720; +} + +message AllRolesResponse { + // Role + repeated Role result = 1; +} + +message AllRolesStreamResponse { + // Role + Role result = 1; +} + +message CreateRoleReq { + Role body = 3704350; +} + +message CreateRoleResponse { + // Role + Role result = 1; +} + +message SearchRolesReq { + // Requested fields. + string fields = 453472492; + // Number of results to return (used with `offset`). + int64 limit = 110364603; + // Number of results to skip before returning any (used with `limit`). + int64 offset = 438591449; + // Fields to sort by. + string sorts = 109773781; + // Match role id. + string id = 3205; + // Match role name. + string name = 3116757; + // Match roles by built_in status. + bool built_in = 35565638; + // Combine given search criteria in a boolean OR expression. + bool filter_or = 515276450; +} + +message SearchRolesResponse { + // Role + repeated Role result = 1; +} + +message SearchRolesStreamResponse { + // Role + Role result = 1; +} + +message SearchRolesWithUserCountReq { + // Requested fields. + string fields = 453472492; + // Number of results to return (used with `offset`). + int64 limit = 110364603; + // Number of results to skip before returning any (used with `limit`). + int64 offset = 438591449; + // Fields to sort by. + string sorts = 109773781; + // Match role id. + string id = 3205; + // Match role name. + string name = 3116757; + // Match roles by built_in status. + bool built_in = 35565638; + // Combine given search criteria in a boolean OR expression. + bool filter_or = 515276450; +} + +message SearchRolesWithUserCountResponse { + // Role + repeated RoleSearch result = 1; +} + +message SearchRolesWithUserCountStreamResponse { + // Role + RoleSearch result = 1; +} + +message RoleReq { + // id of role + string role_id = 413241617; +} + +message RoleResponse { + // Role + Role result = 1; +} + +message UpdateRoleReq { + // id of role + string role_id = 413241617; + Role body = 3704350; +} + +message UpdateRoleResponse { + // Role + Role result = 1; +} + +message DeleteRoleReq { + // id of role + string role_id = 413241617; +} + +message DeleteRoleResponse { + // Successfully deleted. + string result = 1; +} + +message RoleGroupsReq { + // id of role + string role_id = 413241617; + // Requested fields. + string fields = 453472492; +} + +message RoleGroupsResponse { + // Groups with role. + repeated Group result = 1; +} + +message RoleGroupsStreamResponse { + // Groups with role. + Group result = 1; +} + +message SetRoleGroupsReq { + // id of role + string role_id = 413241617; + repeated string body = 3704350; +} + +message SetRoleGroupsResponse { + // Groups with role. + repeated Group result = 1; +} + +message SetRoleGroupsStreamResponse { + // Groups with role. + Group result = 1; +} + +message RoleUsersReq { + // id of role + string role_id = 413241617; + // Requested fields. + string fields = 453472492; + // Get only users associated directly with the role: exclude those only associated through groups. + bool direct_association_only = 284024400; +} + +message RoleUsersResponse { + // Users with role. + repeated User result = 1; +} + +message RoleUsersStreamResponse { + // Users with role. + User result = 1; +} + +message SetRoleUsersReq { + // id of role + string role_id = 413241617; + repeated string body = 3704350; +} + +message SetRoleUsersResponse { + // Users with role. + repeated User result = 1; +} + +message SetRoleUsersStreamResponse { + // Users with role. + User result = 1; +} + +message ScheduledPlansForSpaceReq { + // Space Id + string space_id = 297928564; + // Requested fields. + string fields = 453472492; +} + +message ScheduledPlansForSpaceResponse { + // Scheduled Plan + repeated ScheduledPlan result = 1; +} + +message ScheduledPlansForSpaceStreamResponse { + // Scheduled Plan + ScheduledPlan result = 1; +} + +message ScheduledPlanReq { + // Scheduled Plan Id + string scheduled_plan_id = 303270096; + // Requested fields. + string fields = 453472492; +} + +message ScheduledPlanResponse { + // Scheduled Plan + ScheduledPlan result = 1; +} + +message UpdateScheduledPlanReq { + // Scheduled Plan Id + string scheduled_plan_id = 303270096; + ScheduledPlan body = 3704350; +} + +message UpdateScheduledPlanResponse { + // Scheduled Plan + ScheduledPlan result = 1; +} + +message DeleteScheduledPlanReq { + // Scheduled Plan Id + string scheduled_plan_id = 303270096; +} + +message DeleteScheduledPlanResponse { + // Successfully deleted. + string result = 1; +} + +message AllScheduledPlansReq { + // Return scheduled plans belonging to this user_id. If not provided, returns scheduled plans owned by the caller. + string user_id = 413336787; + // Comma delimited list of field names. If provided, only the fields specified will be included in the response + string fields = 453472492; + // Return scheduled plans belonging to all users (caller needs see_schedules permission) + bool all_users = 508754373; +} + +message AllScheduledPlansResponse { + // Scheduled Plan + repeated ScheduledPlan result = 1; +} + +message AllScheduledPlansStreamResponse { + // Scheduled Plan + ScheduledPlan result = 1; +} + +message CreateScheduledPlanReq { + ScheduledPlan body = 3704350; +} + +message CreateScheduledPlanResponse { + // Scheduled Plan + ScheduledPlan result = 1; +} + +message ScheduledPlanRunOnceReq { + ScheduledPlan body = 3704350; +} + +message ScheduledPlanRunOnceResponse { + // Scheduled Plan + ScheduledPlan result = 1; +} + +message ScheduledPlansForLookReq { + // Look Id + string look_id = 413287022; + // User Id (default is requesting user if not specified) + string user_id = 413336787; + // Requested fields. + string fields = 453472492; + // Return scheduled plans belonging to all users for the look + bool all_users = 508754373; +} + +message ScheduledPlansForLookResponse { + // Scheduled Plan + repeated ScheduledPlan result = 1; +} + +message ScheduledPlansForLookStreamResponse { + // Scheduled Plan + ScheduledPlan result = 1; +} + +message ScheduledPlansForDashboardReq { + // Dashboard Id + string dashboard_id = 311679049; + // User Id (default is requesting user if not specified) + string user_id = 413336787; + // Return scheduled plans belonging to all users for the dashboard + bool all_users = 508754373; + // Requested fields. + string fields = 453472492; +} + +message ScheduledPlansForDashboardResponse { + // Scheduled Plan + repeated ScheduledPlan result = 1; +} + +message ScheduledPlansForDashboardStreamResponse { + // Scheduled Plan + ScheduledPlan result = 1; +} + +message ScheduledPlansForLookmlDashboardReq { + // LookML Dashboard Id + string lookml_dashboard_id = 351003409; + // User Id (default is requesting user if not specified) + string user_id = 413336787; + // Requested fields. + string fields = 453472492; + // Return scheduled plans belonging to all users for the dashboard + bool all_users = 508754373; +} + +message ScheduledPlansForLookmlDashboardResponse { + // Scheduled Plan + repeated ScheduledPlan result = 1; +} + +message ScheduledPlansForLookmlDashboardStreamResponse { + // Scheduled Plan + ScheduledPlan result = 1; +} + +message ScheduledPlanRunOnceByIdReq { + // Id of schedule plan to copy and run + string scheduled_plan_id = 303270096; + WriteScheduledPlan body = 3704350; +} + +message ScheduledPlanRunOnceByIdResponse { + // Scheduled Plan + ScheduledPlan result = 1; +} + +message SessionReq { + +} + +message SessionResponse { + // Session + ApiSession result = 1; +} + +message UpdateSessionReq { + ApiSession body = 3704350; +} + +message UpdateSessionResponse { + // Session + ApiSession result = 1; +} + +message AllThemesReq { + // Requested fields. + string fields = 453472492; +} + +message AllThemesResponse { + // Themes + repeated Theme result = 1; +} + +message AllThemesStreamResponse { + // Themes + Theme result = 1; +} + +message CreateThemeReq { + Theme body = 3704350; +} + +message CreateThemeResponse { + // Theme + Theme result = 1; +} + +message SearchThemesReq { + // Match theme id. + string id = 3205; + // Match theme name. + string name = 3116757; + // Timestamp for activation. + google.protobuf.Timestamp begin_at = 489666644; + // Timestamp for expiration. + google.protobuf.Timestamp end_at = 440737243; + // Number of results to return (used with `offset`). + int64 limit = 110364603; + // Number of results to skip before returning any (used with `limit`). + int64 offset = 438591449; + // Fields to sort by. + string sorts = 109773781; + // Requested fields. + string fields = 453472492; + // Combine given search criteria in a boolean OR expression + bool filter_or = 515276450; +} + +message SearchThemesResponse { + // Themes + repeated Theme result = 1; +} + +message SearchThemesStreamResponse { + // Themes + Theme result = 1; +} + +message DefaultThemeReq { + // Timestamp representing the target datetime for the active period. Defaults to 'now' + google.protobuf.Timestamp ts = 3681; +} + +message DefaultThemeResponse { + // Theme + Theme result = 1; +} + +message SetDefaultThemeReq { + // Name of theme to set as default + string name = 3116757; +} + +message SetDefaultThemeResponse { + // Theme + Theme result = 1; +} + +message ActiveThemesReq { + // Name of theme + string name = 3116757; + // Timestamp representing the target datetime for the active period. Defaults to 'now' + google.protobuf.Timestamp ts = 3681; + // Requested fields. + string fields = 453472492; +} + +message ActiveThemesResponse { + // Themes + repeated Theme result = 1; +} + +message ActiveThemesStreamResponse { + // Themes + Theme result = 1; +} + +message ThemeOrDefaultReq { + // Name of theme + string name = 3116757; + // Timestamp representing the target datetime for the active period. Defaults to 'now' + google.protobuf.Timestamp ts = 3681; +} + +message ThemeOrDefaultResponse { + // Theme + Theme result = 1; +} + +message ValidateThemeReq { + Theme body = 3704350; +} + +message ValidateThemeResponse { + +} + +message ThemeReq { + // Id of theme + string theme_id = 297627057; + // Requested fields. + string fields = 453472492; +} + +message ThemeResponse { + // Theme + Theme result = 1; +} + +message UpdateThemeReq { + // Id of theme + string theme_id = 297627057; + Theme body = 3704350; +} + +message UpdateThemeResponse { + // Theme + Theme result = 1; +} + +message DeleteThemeReq { + // Id of theme + string theme_id = 297627057; +} + +message DeleteThemeResponse { + // Successfully deleted. + string result = 1; +} + +message SearchCredentialsEmailReq { + // Requested fields. + string fields = 453472492; + // Number of results to return (used with `offset`). + int64 limit = 110364603; + // Number of results to skip before returning any (used with `limit`). + int64 offset = 438591449; + // Fields to sort by. + string sorts = 109773781; + // Match credentials_email id. + string id = 3205; + // Match credentials_email email. + string email = 102965020; + // Find credentials_email that match given emails. + string emails = 449824955; + // Combine given search criteria in a boolean OR expression. + bool filter_or = 515276450; +} + +message SearchCredentialsEmailResponse { + // Credentials Email + repeated CredentialsEmailSearch result = 1; +} + +message SearchCredentialsEmailStreamResponse { + // Credentials Email + CredentialsEmailSearch result = 1; +} + +message MeReq { + // Requested fields. + string fields = 453472492; +} + +message MeResponse { + // Current user. + User result = 1; +} + +message AllUsersReq { + // Requested fields. + string fields = 453472492; + // DEPRECATED. Use limit and offset instead. Return only page N of paginated results + int64 page = 3110993; + // DEPRECATED. Use limit and offset instead. Return N rows of data per page + int64 per_page = 262811729; + // Number of results to return. (used with offset and takes priority over page and per_page) + int64 limit = 110364603; + // Number of results to skip before returning any. (used with limit and takes priority over page and per_page) + int64 offset = 438591449; + // Fields to sort by. + string sorts = 109773781; + // Optional list of ids to get specific users. + string ids = 113720; +} + +message AllUsersResponse { + // All users. + repeated User result = 1; +} + +message AllUsersStreamResponse { + // All users. + User result = 1; +} + +message CreateUserReq { + User body = 3704350; + // Requested fields. + string fields = 453472492; +} + +message CreateUserResponse { + // Created User + User result = 1; +} + +message SearchUsersReq { + // Include only these fields in the response + string fields = 453472492; + // DEPRECATED. Use limit and offset instead. Return only page N of paginated results + int64 page = 3110993; + // DEPRECATED. Use limit and offset instead. Return N rows of data per page + int64 per_page = 262811729; + // Number of results to return. (used with offset and takes priority over page and per_page) + int64 limit = 110364603; + // Number of results to skip before returning any. (used with limit and takes priority over page and per_page) + int64 offset = 438591449; + // Fields to sort by. + string sorts = 109773781; + // Match User Id. + string id = 3205; + // Match First name. + string first_name = 277109009; + // Match Last name. + string last_name = 510613627; + // Search for user accounts associated with Looker employees + bool verified_looker_employee = 397997346; + // Search for only embed users + bool embed_user = 479154120; + // Search for the user with this email address + string email = 102965020; + // Search for disabled user accounts + bool is_disabled = 332107427; + // Combine given search criteria in a boolean OR expression + bool filter_or = 515276450; + // Search for users who have access to this content_metadata item + string content_metadata_id = 4111259; + // Search for users who are direct members of this group + string group_id = 287220091; +} + +message SearchUsersResponse { + // Matching users. + repeated User result = 1; +} + +message SearchUsersStreamResponse { + // Matching users. + User result = 1; +} + +message SearchUsersNamesReq { + // Pattern to match + string pattern = 523310252; + // Include only these fields in the response + string fields = 453472492; + // DEPRECATED. Use limit and offset instead. Return only page N of paginated results + int64 page = 3110993; + // DEPRECATED. Use limit and offset instead. Return N rows of data per page + int64 per_page = 262811729; + // Number of results to return. (used with offset and takes priority over page and per_page) + int64 limit = 110364603; + // Number of results to skip before returning any. (used with limit and takes priority over page and per_page) + int64 offset = 438591449; + // Fields to sort by + string sorts = 109773781; + // Match User Id + string id = 3205; + // Match First name + string first_name = 277109009; + // Match Last name + string last_name = 510613627; + // Match Verified Looker employee + bool verified_looker_employee = 397997346; + // Match Email Address + string email = 102965020; + // Include or exclude disabled accounts in the results + bool is_disabled = 332107427; +} + +message SearchUsersNamesResponse { + // Matching users. + repeated User result = 1; +} + +message SearchUsersNamesStreamResponse { + // Matching users. + User result = 1; +} + +message UserReq { + // Id of user + string user_id = 413336787; + // Requested fields. + string fields = 453472492; +} + +message UserResponse { + // Specified user. + User result = 1; +} + +message UpdateUserReq { + // Id of user + string user_id = 413336787; + User body = 3704350; + // Requested fields. + string fields = 453472492; +} + +message UpdateUserResponse { + // New state for specified user. + User result = 1; +} + +message DeleteUserReq { + // Id of user + string user_id = 413336787; +} + +message DeleteUserResponse { + // User successfully deleted. + string result = 1; +} + +message UserForCredentialReq { + // Type name of credential + string credential_type = 335279800; + // Id of credential + string credential_id = 318409473; + // Requested fields. + string fields = 453472492; +} + +message UserForCredentialResponse { + // Specified user. + User result = 1; +} + +message UserCredentialsEmailReq { + // Id of user + string user_id = 413336787; + // Requested fields. + string fields = 453472492; +} + +message UserCredentialsEmailResponse { + // Email/Password Credential + CredentialsEmail result = 1; +} + +message CreateUserCredentialsEmailReq { + // Id of user + string user_id = 413336787; + CredentialsEmail body = 3704350; + // Requested fields. + string fields = 453472492; +} + +message CreateUserCredentialsEmailResponse { + // Email/Password Credential + CredentialsEmail result = 1; +} + +message UpdateUserCredentialsEmailReq { + // Id of user + string user_id = 413336787; + CredentialsEmail body = 3704350; + // Requested fields. + string fields = 453472492; +} + +message UpdateUserCredentialsEmailResponse { + // Email/Password Credential + CredentialsEmail result = 1; +} + +message DeleteUserCredentialsEmailReq { + // Id of user + string user_id = 413336787; +} + +message DeleteUserCredentialsEmailResponse { + // Successfully deleted. + string result = 1; +} + +message UserCredentialsTotpReq { + // Id of user + string user_id = 413336787; + // Requested fields. + string fields = 453472492; +} + +message UserCredentialsTotpResponse { + // Two-Factor Credential + CredentialsTotp result = 1; +} + +message CreateUserCredentialsTotpReq { + // Id of user + string user_id = 413336787; + CredentialsTotp body = 3704350; + // Requested fields. + string fields = 453472492; +} + +message CreateUserCredentialsTotpResponse { + // Two-Factor Credential + CredentialsTotp result = 1; +} + +message DeleteUserCredentialsTotpReq { + // Id of user + string user_id = 413336787; +} + +message DeleteUserCredentialsTotpResponse { + // Successfully deleted. + string result = 1; +} + +message UserCredentialsLdapReq { + // Id of user + string user_id = 413336787; + // Requested fields. + string fields = 453472492; +} + +message UserCredentialsLdapResponse { + // LDAP Credential + CredentialsLDAP result = 1; +} + +message DeleteUserCredentialsLdapReq { + // Id of user + string user_id = 413336787; +} + +message DeleteUserCredentialsLdapResponse { + // Successfully deleted. + string result = 1; +} + +message UserCredentialsGoogleReq { + // Id of user + string user_id = 413336787; + // Requested fields. + string fields = 453472492; +} + +message UserCredentialsGoogleResponse { + // Google Auth Credential + CredentialsGoogle result = 1; +} + +message DeleteUserCredentialsGoogleReq { + // Id of user + string user_id = 413336787; +} + +message DeleteUserCredentialsGoogleResponse { + // Successfully deleted. + string result = 1; +} + +message UserCredentialsSamlReq { + // Id of user + string user_id = 413336787; + // Requested fields. + string fields = 453472492; +} + +message UserCredentialsSamlResponse { + // Saml Auth Credential + CredentialsSaml result = 1; +} + +message DeleteUserCredentialsSamlReq { + // Id of user + string user_id = 413336787; +} + +message DeleteUserCredentialsSamlResponse { + // Successfully deleted. + string result = 1; +} + +message UserCredentialsOidcReq { + // Id of user + string user_id = 413336787; + // Requested fields. + string fields = 453472492; +} + +message UserCredentialsOidcResponse { + // OIDC Auth Credential + CredentialsOIDC result = 1; +} + +message DeleteUserCredentialsOidcReq { + // Id of user + string user_id = 413336787; +} + +message DeleteUserCredentialsOidcResponse { + // Successfully deleted. + string result = 1; +} + +message UserCredentialsApi3Req { + // Id of user + string user_id = 413336787; + // Id of API 3 Credential + string credentials_api3_id = 409889023; + // Requested fields. + string fields = 453472492; +} + +message UserCredentialsApi3Response { + // API 3 Credential + CredentialsApi3 result = 1; +} + +message DeleteUserCredentialsApi3Req { + // Id of user + string user_id = 413336787; + // Id of API 3 Credential + string credentials_api3_id = 409889023; +} + +message DeleteUserCredentialsApi3Response { + // Successfully deleted. + string result = 1; +} + +message AllUserCredentialsApi3sReq { + // Id of user + string user_id = 413336787; + // Requested fields. + string fields = 453472492; +} + +message AllUserCredentialsApi3sResponse { + // API 3 Credential + repeated CredentialsApi3 result = 1; +} + +message AllUserCredentialsApi3sStreamResponse { + // API 3 Credential + CredentialsApi3 result = 1; +} + +message CreateUserCredentialsApi3Req { + // Id of user + string user_id = 413336787; + // Requested fields. + string fields = 453472492; +} + +message CreateUserCredentialsApi3Response { + // API 3 Credential + CreateCredentialsApi3 result = 1; +} + +message UserCredentialsEmbedReq { + // Id of user + string user_id = 413336787; + // Id of Embedding Credential + string credentials_embed_id = 285067495; + // Requested fields. + string fields = 453472492; +} + +message UserCredentialsEmbedResponse { + // Embedding Credential + CredentialsEmbed result = 1; +} + +message DeleteUserCredentialsEmbedReq { + // Id of user + string user_id = 413336787; + // Id of Embedding Credential + string credentials_embed_id = 285067495; +} + +message DeleteUserCredentialsEmbedResponse { + // Successfully deleted. + string result = 1; +} + +message AllUserCredentialsEmbedsReq { + // Id of user + string user_id = 413336787; + // Requested fields. + string fields = 453472492; +} + +message AllUserCredentialsEmbedsResponse { + // Embedding Credential + repeated CredentialsEmbed result = 1; +} + +message AllUserCredentialsEmbedsStreamResponse { + // Embedding Credential + CredentialsEmbed result = 1; +} + +message UserCredentialsLookerOpenidReq { + // Id of user + string user_id = 413336787; + // Requested fields. + string fields = 453472492; +} + +message UserCredentialsLookerOpenidResponse { + // Looker OpenId Credential + CredentialsLookerOpenid result = 1; +} + +message DeleteUserCredentialsLookerOpenidReq { + // Id of user + string user_id = 413336787; +} + +message DeleteUserCredentialsLookerOpenidResponse { + // Successfully deleted. + string result = 1; +} + +message UserSessionReq { + // Id of user + string user_id = 413336787; + // Id of Web Login Session + string session_id = 300521991; + // Requested fields. + string fields = 453472492; +} + +message UserSessionResponse { + // Web Login Session + Session result = 1; +} + +message DeleteUserSessionReq { + // Id of user + string user_id = 413336787; + // Id of Web Login Session + string session_id = 300521991; +} + +message DeleteUserSessionResponse { + // Successfully deleted. + string result = 1; +} + +message AllUserSessionsReq { + // Id of user + string user_id = 413336787; + // Requested fields. + string fields = 453472492; +} + +message AllUserSessionsResponse { + // Web Login Session + repeated Session result = 1; +} + +message AllUserSessionsStreamResponse { + // Web Login Session + Session result = 1; +} + +message CreateUserCredentialsEmailPasswordResetReq { + // Id of user + string user_id = 413336787; + // Expiring token. + bool expires = 495943341; + // Requested fields. + string fields = 453472492; +} + +message CreateUserCredentialsEmailPasswordResetResponse { + // email/password credential + CredentialsEmail result = 1; +} + +message UserRolesReq { + // Id of user + string user_id = 413336787; + // Requested fields. + string fields = 453472492; + // Get only roles associated directly with the user: exclude those only associated through groups. + bool direct_association_only = 284024400; +} + +message UserRolesResponse { + // Roles of user. + repeated Role result = 1; +} + +message UserRolesStreamResponse { + // Roles of user. + Role result = 1; +} + +message SetUserRolesReq { + // Id of user + string user_id = 413336787; + repeated string body = 3704350; + // Requested fields. + string fields = 453472492; +} + +message SetUserRolesResponse { + // Roles of user. + repeated Role result = 1; +} + +message SetUserRolesStreamResponse { + // Roles of user. + Role result = 1; +} + +message UserAttributeUserValuesReq { + // Id of user + string user_id = 413336787; + // Requested fields. + string fields = 453472492; + // Specific user attributes to request. Omit or leave blank to request all user attributes. + string user_attribute_ids = 329947583; + // If true, returns all values in the search path instead of just the first value found. Useful for debugging group precedence. + bool all_values = 322013600; + // If true, returns an empty record for each requested attribute that has no user, group, or default value. + bool include_unset = 346429271; +} + +message UserAttributeUserValuesResponse { + // Value of user attribute. + repeated UserAttributeWithValue result = 1; +} + +message UserAttributeUserValuesStreamResponse { + // Value of user attribute. + UserAttributeWithValue result = 1; +} + +message SetUserAttributeUserValueReq { + // Id of user + string user_id = 413336787; + // Id of user attribute + string user_attribute_id = 18993938; + UserAttributeWithValue body = 3704350; +} + +message SetUserAttributeUserValueResponse { + // User attribute value. + UserAttributeWithValue result = 1; +} + +message DeleteUserAttributeUserValueReq { + // Id of user + string user_id = 413336787; + // Id of user attribute + string user_attribute_id = 18993938; +} + +message DeleteUserAttributeUserValueResponse { + // Deleted +} + +message SendUserCredentialsEmailPasswordResetReq { + // Id of user + string user_id = 413336787; + // Requested fields. + string fields = 453472492; +} + +message SendUserCredentialsEmailPasswordResetResponse { + // email/password credential + CredentialsEmail result = 1; +} + +message WipeoutUserEmailsReq { + // Id of user + string user_id = 413336787; + UserEmailOnly body = 3704350; + // Requested fields. + string fields = 453472492; +} + +message WipeoutUserEmailsResponse { + // New state for specified user. + User result = 1; +} + +message CreateEmbedUserReq { + CreateEmbedUserRequest body = 3704350; +} + +message CreateEmbedUserResponse { + // Created embed user + UserPublic result = 1; +} + +message AllUserAttributesReq { + // Requested fields. + string fields = 453472492; + // Fields to order the results by. Sortable fields include: name, label + string sorts = 109773781; +} + +message AllUserAttributesResponse { + // User Attribute + repeated UserAttribute result = 1; +} + +message AllUserAttributesStreamResponse { + // User Attribute + UserAttribute result = 1; +} + +message CreateUserAttributeReq { + UserAttribute body = 3704350; + // Requested fields. + string fields = 453472492; +} + +message CreateUserAttributeResponse { + // User Attribute + UserAttribute result = 1; +} + +message UserAttributeReq { + // Id of user attribute + string user_attribute_id = 18993938; + // Requested fields. + string fields = 453472492; +} + +message UserAttributeResponse { + // User Attribute + UserAttribute result = 1; +} + +message UpdateUserAttributeReq { + // Id of user attribute + string user_attribute_id = 18993938; + UserAttribute body = 3704350; + // Requested fields. + string fields = 453472492; +} + +message UpdateUserAttributeResponse { + // User Attribute + UserAttribute result = 1; +} + +message DeleteUserAttributeReq { + // Id of user attribute + string user_attribute_id = 18993938; +} + +message DeleteUserAttributeResponse { + // Successfully deleted. + string result = 1; +} + +message AllUserAttributeGroupValuesReq { + // Id of user attribute + string user_attribute_id = 18993938; + // Requested fields. + string fields = 453472492; +} + +message AllUserAttributeGroupValuesResponse { + // All group values for attribute. + repeated UserAttributeGroupValue result = 1; +} + +message AllUserAttributeGroupValuesStreamResponse { + // All group values for attribute. + UserAttributeGroupValue result = 1; +} + +message SetUserAttributeGroupValuesReq { + // Id of user attribute + string user_attribute_id = 18993938; + repeated UserAttributeGroupValue body = 3704350; +} + +message SetUserAttributeGroupValuesResponse { + // Array of group values. + repeated UserAttributeGroupValue result = 1; +} + +message SetUserAttributeGroupValuesStreamResponse { + // Array of group values. + UserAttributeGroupValue result = 1; +} + +message AllWorkspacesReq { + +} + +message AllWorkspacesResponse { + // Workspace + repeated Workspace result = 1; +} + +message AllWorkspacesStreamResponse { + // Workspace + Workspace result = 1; +} + +message WorkspaceReq { + // Id of the workspace + string workspace_id = 456431629; +} + +message WorkspaceResponse { + // Workspace + Workspace result = 1; +} diff --git a/proto/grpc_proxy/src/main/proto/sdk/streams.proto b/proto/grpc_proxy/src/main/proto/sdk/streams.proto new file mode 100644 index 000000000..87788fe68 --- /dev/null +++ b/proto/grpc_proxy/src/main/proto/sdk/streams.proto @@ -0,0 +1,3684 @@ +// MIT License +// +// Copyright (c) 2021 Looker Data Sciences, Inc. +// +// Permission is hereby granted, free of charge, to any person obtaining a copy +// of this software and associated documentation files (the "Software"), to deal +// in the Software without restriction, including without limitation the rights +// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +// copies of the Software, and to permit persons to whom the Software is +// furnished to do so, subject to the following conditions: +// +// The above copyright notice and this permission notice shall be included in all +// copies or substantial portions of the Software. +// +// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +// SOFTWARE. +// + +// 438 API methods + + +syntax = "proto3"; + +package looker; + +option java_package = "com.google.looker.grpc.services"; +option java_multiple_files = true; + +import 'sdk/models.proto'; + +service LookerStreamingService { + + // Alert: Alert + + // ### Search Alerts + // + rpc SearchAlerts(SearchAlertsReq) returns (stream SearchAlertsStreamResponse); + + // ### Get an alert by a given alert ID + // + rpc GetAlert(GetAlertReq) returns (stream GetAlertResponse); + + // ### Update an alert + // # Required fields: `owner_id`, `field`, `destinations`, `comparison_type`, `threshold`, `cron` + // # + // + rpc UpdateAlert(UpdateAlertReq) returns (stream UpdateAlertResponse); + + // ### Update select alert fields + // # Available fields: `owner_id`, `is_disabled`, `disabled_reason`, `is_public`, `threshold` + // # + // + rpc UpdateAlertField(UpdateAlertFieldReq) returns (stream UpdateAlertFieldResponse); + + // ### Delete an alert by a given alert ID + // + rpc DeleteAlert(DeleteAlertReq) returns (stream DeleteAlertResponse); + + // ### Create a new alert and return details of the newly created object + // + // Required fields: `field`, `destinations`, `comparison_type`, `threshold`, `cron` + // + // Example Request: + // Run alert on dashboard element '103' at 5am every day. Send an email to 'test@test.com' if inventory for Los Angeles (using dashboard filter `Warehouse Name`) is lower than 1,000 + // ``` + // { + // "cron": "0 5 * * *", + // "custom_title": "Alert when LA inventory is low", + // "dashboard_element_id": 103, + // "applied_dashboard_filters": [ + // { + // "filter_title": "Warehouse Name", + // "field_name": "distribution_centers.name", + // "filter_value": "Los Angeles CA", + // "filter_description": "is Los Angeles CA" + // } + // ], + // "comparison_type": "LESS_THAN", + // "destinations": [ + // { + // "destination_type": "EMAIL", + // "email_address": "test@test.com" + // } + // ], + // "field": { + // "title": "Number on Hand", + // "name": "inventory_items.number_on_hand" + // }, + // "is_disabled": false, + // "is_public": true, + // "threshold": 1000 + // } + // ``` + // + rpc CreateAlert(CreateAlertReq) returns (stream CreateAlertResponse); + + // ### Enqueue an Alert by ID + // + rpc EnqueueAlert(EnqueueAlertReq) returns (stream EnqueueAlertResponse); + + + + // ApiAuth: API Authentication + + // ### Present client credentials to obtain an authorization token + // + // Looker API implements the OAuth2 [Resource Owner Password Credentials Grant](https://looker.com/docs/r/api/outh2_resource_owner_pc) pattern. + // The client credentials required for this login must be obtained by creating an API3 key on a user account + // in the Looker Admin console. The API3 key consists of a public `client_id` and a private `client_secret`. + // + // The access token returned by `login` must be used in the HTTP Authorization header of subsequent + // API requests, like this: + // ``` + // Authorization: token 4QDkCyCtZzYgj4C2p2cj3csJH7zqS5RzKs2kTnG4 + // ``` + // Replace "4QDkCy..." with the `access_token` value returned by `login`. + // The word `token` is a string literal and must be included exactly as shown. + // + // This function can accept `client_id` and `client_secret` parameters as URL query params or as www-form-urlencoded params in the body of the HTTP request. Since there is a small risk that URL parameters may be visible to intermediate nodes on the network route (proxies, routers, etc), passing credentials in the body of the request is considered more secure than URL params. + // + // Example of passing credentials in the HTTP request body: + // ```` + // POST HTTP /login + // Content-Type: application/x-www-form-urlencoded + // + // client_id=CGc9B7v7J48dQSJvxxx&client_secret=nNVS9cSS3xNpSC9JdsBvvvvv + // ```` + // + // ### Best Practice: + // Always pass credentials in body params. Pass credentials in URL query params **only** when you cannot pass body params due to application, tool, or other limitations. + // + // For more information and detailed examples of Looker API authorization, see [How to Authenticate to Looker API3](https://github.com/looker/looker-sdk-ruby/blob/master/authentication.md). + // + rpc Login(LoginReq) returns (stream LoginResponse); + + // ### Create an access token that runs as a given user. + // + // This can only be called by an authenticated admin user. It allows that admin to generate a new + // authentication token for the user with the given user id. That token can then be used for subsequent + // API calls - which are then performed *as* that target user. + // + // The target user does *not* need to have a pre-existing API client_id/client_secret pair. And, no such + // credentials are created by this call. + // + // This allows for building systems where api user authentication for an arbitrary number of users is done + // outside of Looker and funneled through a single 'service account' with admin permissions. Note that a + // new access token is generated on each call. If target users are going to be making numerous API + // calls in a short period then it is wise to cache this authentication token rather than call this before + // each of those API calls. + // + // See 'login' for more detail on the access token and how to use it. + // + rpc LoginUser(LoginUserReq) returns (stream LoginUserResponse); + + // ### Logout of the API and invalidate the current access token. + // + rpc Logout(LogoutReq) returns (stream LogoutResponse); + + + + // Auth: Manage User Authentication Configuration + + // ### Create an embed secret using the specified information. + // + // The value of the `secret` field will be set by Looker and returned. + // + rpc CreateEmbedSecret(CreateEmbedSecretReq) returns (stream CreateEmbedSecretResponse); + + // ### Delete an embed secret. + // + rpc DeleteEmbedSecret(DeleteEmbedSecretReq) returns (stream DeleteEmbedSecretResponse); + + // ### Create SSO Embed URL + // + // Creates an SSO embed URL and cryptographically signs it with an embed secret. + // This signed URL can then be used to instantiate a Looker embed session in a PBL web application. + // Do not make any modifications to this URL - any change may invalidate the signature and + // cause the URL to fail to load a Looker embed session. + // + // A signed SSO embed URL can only be used once. After it has been used to request a page from the + // Looker server, the URL is invalid. Future requests using the same URL will fail. This is to prevent + // 'replay attacks'. + // + // The `target_url` property must be a complete URL of a Looker UI page - scheme, hostname, path and query params. + // To load a dashboard with id 56 and with a filter of `Date=1 years`, the looker URL would look like `https:/myname.looker.com/dashboards/56?Date=1%20years`. + // The best way to obtain this target_url is to navigate to the desired Looker page in your web browser, + // copy the URL shown in the browser address bar and paste it into the `target_url` property as a quoted string value in this API request. + // + // Permissions for the embed user are defined by the groups in which the embed user is a member (group_ids property) + // and the lists of models and permissions assigned to the embed user. + // At a minimum, you must provide values for either the group_ids property, or both the models and permissions properties. + // These properties are additive; an embed user can be a member of certain groups AND be granted access to models and permissions. + // + // The embed user's access is the union of permissions granted by the group_ids, models, and permissions properties. + // + // This function does not strictly require all group_ids, user attribute names, or model names to exist at the moment the + // SSO embed url is created. Unknown group_id, user attribute names or model names will be passed through to the output URL. + // To diagnose potential problems with an SSO embed URL, you can copy the signed URL into the Embed URI Validator text box in `/admin/embed`. + // + // The `secret_id` parameter is optional. If specified, its value must be the id of an active secret defined in the Looker instance. + // if not specified, the URL will be signed using the newest active secret defined in the Looker instance. + // + // #### Security Note + // Protect this signed URL as you would an access token or password credentials - do not write + // it to disk, do not pass it to a third party, and only pass it through a secure HTTPS + // encrypted transport. + // + rpc CreateSsoEmbedUrl(CreateSsoEmbedUrlReq) returns (stream CreateSsoEmbedUrlResponse); + + // ### Create an Embed URL + // + // Creates an embed URL that runs as the Looker user making this API call. ("Embed as me") + // This embed URL can then be used to instantiate a Looker embed session in a + // "Powered by Looker" (PBL) web application. + // + // This is similar to Private Embedding (https://docs.looker.com/r/admin/embed/private-embed). Instead of + // of logging into the Web UI to authenticate, the user has already authenticated against the API to be able to + // make this call. However, unlike Private Embed where the user has access to any other part of the Looker UI, + // the embed web session created by requesting the EmbedUrlResponse.url in a browser only has access to + // content visible under the `/embed` context. + // + // An embed URL can only be used once, and must be used within 5 minutes of being created. After it + // has been used to request a page from the Looker server, the URL is invalid. Future requests using + // the same URL will fail. This is to prevent 'replay attacks'. + // + // The `target_url` property must be a complete URL of a Looker Embedded UI page - scheme, hostname, path starting with "/embed" and query params. + // To load a dashboard with id 56 and with a filter of `Date=1 years`, the looker Embed URL would look like `https://myname.looker.com/embed/dashboards/56?Date=1%20years`. + // The best way to obtain this target_url is to navigate to the desired Looker page in your web browser, + // copy the URL shown in the browser address bar, insert "/embed" after the host/port, and paste it into the `target_url` property as a quoted string value in this API request. + // + // #### Security Note + // Protect this embed URL as you would an access token or password credentials - do not write + // it to disk, do not pass it to a third party, and only pass it through a secure HTTPS + // encrypted transport. + // + rpc CreateEmbedUrlAsMe(CreateEmbedUrlAsMeReq) returns (stream CreateEmbedUrlAsMeResponse); + + // ### Get the LDAP configuration. + // + // Looker can be optionally configured to authenticate users against an Active Directory or other LDAP directory server. + // LDAP setup requires coordination with an administrator of that directory server. + // + // Only Looker administrators can read and update the LDAP configuration. + // + // Configuring LDAP impacts authentication for all users. This configuration should be done carefully. + // + // Looker maintains a single LDAP configuration. It can be read and updated. Updates only succeed if the new state will be valid (in the sense that all required fields are populated); it is up to you to ensure that the configuration is appropriate and correct). + // + // LDAP is enabled or disabled for Looker using the **enabled** field. + // + // Looker will never return an **auth_password** field. That value can be set, but never retrieved. + // + // See the [Looker LDAP docs](https://www.looker.com/docs/r/api/ldap_setup) for additional information. + // + rpc LdapConfig(LdapConfigReq) returns (stream LdapConfigResponse); + + // ### Update the LDAP configuration. + // + // Configuring LDAP impacts authentication for all users. This configuration should be done carefully. + // + // Only Looker administrators can read and update the LDAP configuration. + // + // LDAP is enabled or disabled for Looker using the **enabled** field. + // + // It is **highly** recommended that any LDAP setting changes be tested using the APIs below before being set globally. + // + // See the [Looker LDAP docs](https://www.looker.com/docs/r/api/ldap_setup) for additional information. + // + rpc UpdateLdapConfig(UpdateLdapConfigReq) returns (stream UpdateLdapConfigResponse); + + // ### Test the connection settings for an LDAP configuration. + // + // This tests that the connection is possible given a connection_host and connection_port. + // + // **connection_host** and **connection_port** are required. **connection_tls** is optional. + // + // Example: + // ```json + // { + // "connection_host": "ldap.example.com", + // "connection_port": "636", + // "connection_tls": true + // } + // ``` + // + // No authentication to the LDAP server is attempted. + // + // The active LDAP settings are not modified. + // + rpc TestLdapConfigConnection(TestLdapConfigConnectionReq) returns (stream TestLdapConfigConnectionResponse); + + // ### Test the connection authentication settings for an LDAP configuration. + // + // This tests that the connection is possible and that a 'server' account to be used by Looker can authenticate to the LDAP server given connection and authentication information. + // + // **connection_host**, **connection_port**, and **auth_username**, are required. **connection_tls** and **auth_password** are optional. + // + // Example: + // ```json + // { + // "connection_host": "ldap.example.com", + // "connection_port": "636", + // "connection_tls": true, + // "auth_username": "cn=looker,dc=example,dc=com", + // "auth_password": "secret" + // } + // ``` + // + // Looker will never return an **auth_password**. If this request omits the **auth_password** field, then the **auth_password** value from the active config (if present) will be used for the test. + // + // The active LDAP settings are not modified. + // + // + rpc TestLdapConfigAuth(TestLdapConfigAuthReq) returns (stream TestLdapConfigAuthResponse); + + // ### Test the user authentication settings for an LDAP configuration without authenticating the user. + // + // This test will let you easily test the mapping for user properties and roles for any user without needing to authenticate as that user. + // + // This test accepts a full LDAP configuration along with a username and attempts to find the full info for the user from the LDAP server without actually authenticating the user. So, user password is not required.The configuration is validated before attempting to contact the server. + // + // **test_ldap_user** is required. + // + // The active LDAP settings are not modified. + // + // + rpc TestLdapConfigUserInfo(TestLdapConfigUserInfoReq) returns (stream TestLdapConfigUserInfoResponse); + + // ### Test the user authentication settings for an LDAP configuration. + // + // This test accepts a full LDAP configuration along with a username/password pair and attempts to authenticate the user with the LDAP server. The configuration is validated before attempting the authentication. + // + // Looker will never return an **auth_password**. If this request omits the **auth_password** field, then the **auth_password** value from the active config (if present) will be used for the test. + // + // **test_ldap_user** and **test_ldap_password** are required. + // + // The active LDAP settings are not modified. + // + // + rpc TestLdapConfigUserAuth(TestLdapConfigUserAuthReq) returns (stream TestLdapConfigUserAuthResponse); + + // ### List All OAuth Client Apps + // + // Lists all applications registered to use OAuth2 login with this Looker instance, including + // enabled and disabled apps. + // + // Results are filtered to include only the apps that the caller (current user) + // has permission to see. + // + rpc AllOauthClientApps(AllOauthClientAppsReq) returns (stream AllOauthClientAppsStreamResponse); + + // ### Get Oauth Client App + // + // Returns the registered app client with matching client_guid. + // + rpc OauthClientApp(OauthClientAppReq) returns (stream OauthClientAppResponse); + + // ### Register an OAuth2 Client App + // + // Registers details identifying an external web app or native app as an OAuth2 login client of the Looker instance. + // The app registration must provide a unique client_guid and redirect_uri that the app will present + // in OAuth login requests. If the client_guid and redirect_uri parameters in the login request do not match + // the app details registered with the Looker instance, the request is assumed to be a forgery and is rejected. + // + rpc RegisterOauthClientApp(RegisterOauthClientAppReq) returns (stream RegisterOauthClientAppResponse); + + // ### Update OAuth2 Client App Details + // + // Modifies the details a previously registered OAuth2 login client app. + // + rpc UpdateOauthClientApp(UpdateOauthClientAppReq) returns (stream UpdateOauthClientAppResponse); + + // ### Delete OAuth Client App + // + // Deletes the registration info of the app with the matching client_guid. + // All active sessions and tokens issued for this app will immediately become invalid. + // + // ### Note: this deletion cannot be undone. + // + rpc DeleteOauthClientApp(DeleteOauthClientAppReq) returns (stream DeleteOauthClientAppResponse); + + // ### Invalidate All Issued Tokens + // + // Immediately invalidates all auth codes, sessions, access tokens and refresh tokens issued for + // this app for ALL USERS of this app. + // + rpc InvalidateTokens(InvalidateTokensReq) returns (stream InvalidateTokensResponse); + + // ### Activate an app for a user + // + // Activates a user for a given oauth client app. This indicates the user has been informed that + // the app will have access to the user's looker data, and that the user has accepted and allowed + // the app to use their Looker account. + // + // Activating a user for an app that the user is already activated with returns a success response. + // + rpc ActivateAppUser(ActivateAppUserReq) returns (stream ActivateAppUserResponse); + + // ### Deactivate an app for a user + // + // Deactivate a user for a given oauth client app. All tokens issued to the app for + // this user will be invalid immediately. Before the user can use the app with their + // Looker account, the user will have to read and accept an account use disclosure statement for the app. + // + // Admin users can deactivate other users, but non-admin users can only deactivate themselves. + // + // As with most REST DELETE operations, this endpoint does not return an error if the indicated + // resource (app or user) does not exist or has already been deactivated. + // + rpc DeactivateAppUser(DeactivateAppUserReq) returns (stream DeactivateAppUserResponse); + + // ### Get the OIDC configuration. + // + // Looker can be optionally configured to authenticate users against an OpenID Connect (OIDC) + // authentication server. OIDC setup requires coordination with an administrator of that server. + // + // Only Looker administrators can read and update the OIDC configuration. + // + // Configuring OIDC impacts authentication for all users. This configuration should be done carefully. + // + // Looker maintains a single OIDC configuation. It can be read and updated. Updates only succeed if the new state will be valid (in the sense that all required fields are populated); it is up to you to ensure that the configuration is appropriate and correct). + // + // OIDC is enabled or disabled for Looker using the **enabled** field. + // + rpc OidcConfig(OidcConfigReq) returns (stream OidcConfigResponse); + + // ### Update the OIDC configuration. + // + // Configuring OIDC impacts authentication for all users. This configuration should be done carefully. + // + // Only Looker administrators can read and update the OIDC configuration. + // + // OIDC is enabled or disabled for Looker using the **enabled** field. + // + // It is **highly** recommended that any OIDC setting changes be tested using the APIs below before being set globally. + // + rpc UpdateOidcConfig(UpdateOidcConfigReq) returns (stream UpdateOidcConfigResponse); + + // ### Get a OIDC test configuration by test_slug. + // + rpc OidcTestConfig(OidcTestConfigReq) returns (stream OidcTestConfigResponse); + + // ### Delete a OIDC test configuration. + // + rpc DeleteOidcTestConfig(DeleteOidcTestConfigReq) returns (stream DeleteOidcTestConfigResponse); + + // ### Create a OIDC test configuration. + // + rpc CreateOidcTestConfig(CreateOidcTestConfigReq) returns (stream CreateOidcTestConfigResponse); + + // ### Get password config. + // + rpc PasswordConfig(PasswordConfigReq) returns (stream PasswordConfigResponse); + + // ### Update password config. + // + rpc UpdatePasswordConfig(UpdatePasswordConfigReq) returns (stream UpdatePasswordConfigResponse); + + // ### Force all credentials_email users to reset their login passwords upon their next login. + // + rpc ForcePasswordResetAtNextLoginForAllUsers(ForcePasswordResetAtNextLoginForAllUsersReq) returns (stream ForcePasswordResetAtNextLoginForAllUsersResponse); + + // ### Get the SAML configuration. + // + // Looker can be optionally configured to authenticate users against a SAML authentication server. + // SAML setup requires coordination with an administrator of that server. + // + // Only Looker administrators can read and update the SAML configuration. + // + // Configuring SAML impacts authentication for all users. This configuration should be done carefully. + // + // Looker maintains a single SAML configuation. It can be read and updated. Updates only succeed if the new state will be valid (in the sense that all required fields are populated); it is up to you to ensure that the configuration is appropriate and correct). + // + // SAML is enabled or disabled for Looker using the **enabled** field. + // + rpc SamlConfig(SamlConfigReq) returns (stream SamlConfigResponse); + + // ### Update the SAML configuration. + // + // Configuring SAML impacts authentication for all users. This configuration should be done carefully. + // + // Only Looker administrators can read and update the SAML configuration. + // + // SAML is enabled or disabled for Looker using the **enabled** field. + // + // It is **highly** recommended that any SAML setting changes be tested using the APIs below before being set globally. + // + rpc UpdateSamlConfig(UpdateSamlConfigReq) returns (stream UpdateSamlConfigResponse); + + // ### Get a SAML test configuration by test_slug. + // + rpc SamlTestConfig(SamlTestConfigReq) returns (stream SamlTestConfigResponse); + + // ### Delete a SAML test configuration. + // + rpc DeleteSamlTestConfig(DeleteSamlTestConfigReq) returns (stream DeleteSamlTestConfigResponse); + + // ### Create a SAML test configuration. + // + rpc CreateSamlTestConfig(CreateSamlTestConfigReq) returns (stream CreateSamlTestConfigResponse); + + // ### Parse the given xml as a SAML IdP metadata document and return the result. + // + rpc ParseSamlIdpMetadata(ParseSamlIdpMetadataReq) returns (stream ParseSamlIdpMetadataResponse); + + // ### Fetch the given url and parse it as a SAML IdP metadata document and return the result. + // Note that this requires that the url be public or at least at a location where the Looker instance + // can fetch it without requiring any special authentication. + // + rpc FetchAndParseSamlIdpMetadata(FetchAndParseSamlIdpMetadataReq) returns (stream FetchAndParseSamlIdpMetadataResponse); + + // ### Get session config. + // + rpc SessionConfig(SessionConfigReq) returns (stream SessionConfigResponse); + + // ### Update session config. + // + rpc UpdateSessionConfig(UpdateSessionConfigReq) returns (stream UpdateSessionConfigResponse); + + // ### Get Support Access Allowlist Users + // + // Returns the users that have been added to the Support Access Allowlist + // + rpc GetSupportAccessAllowlistEntries(GetSupportAccessAllowlistEntriesReq) returns (stream GetSupportAccessAllowlistEntriesStreamResponse); + + // ### Add Support Access Allowlist Users + // + // Adds a list of emails to the Allowlist, using the provided reason + // + rpc AddSupportAccessAllowlistEntries(AddSupportAccessAllowlistEntriesReq) returns (stream AddSupportAccessAllowlistEntriesStreamResponse); + + // ### Delete Support Access Allowlist User + // + // Deletes the specified Allowlist Entry Id + // + rpc DeleteSupportAccessAllowlistEntry(DeleteSupportAccessAllowlistEntryReq) returns (stream DeleteSupportAccessAllowlistEntryResponse); + + // ### Enable Support Access + // + // Enables Support Access for the provided duration + // + rpc EnableSupportAccess(EnableSupportAccessReq) returns (stream EnableSupportAccessResponse); + + // ### Disable Support Access + // + // Disables Support Access immediately + // + rpc DisableSupportAccess(DisableSupportAccessReq) returns (stream DisableSupportAccessResponse); + + // ### Support Access Status + // + // Returns the current Support Access Status + // + rpc SupportAccessStatus(SupportAccessStatusReq) returns (stream SupportAccessStatusResponse); + + // ### Get currently locked-out users. + // + rpc AllUserLoginLockouts(AllUserLoginLockoutsReq) returns (stream AllUserLoginLockoutsStreamResponse); + + // ### Search currently locked-out users. + // + rpc SearchUserLoginLockouts(SearchUserLoginLockoutsReq) returns (stream SearchUserLoginLockoutsStreamResponse); + + // ### Removes login lockout for the associated user. + // + rpc DeleteUserLoginLockout(DeleteUserLoginLockoutReq) returns (stream DeleteUserLoginLockoutResponse); + + + + // Board: Manage Boards + + // ### Get information about all boards. + // + rpc AllBoards(AllBoardsReq) returns (stream AllBoardsStreamResponse); + + // ### Create a new board. + // + rpc CreateBoard(CreateBoardReq) returns (stream CreateBoardResponse); + + // ### Search Boards + // + // If multiple search params are given and `filter_or` is FALSE or not specified, + // search params are combined in a logical AND operation. + // Only rows that match *all* search param criteria will be returned. + // + // If `filter_or` is TRUE, multiple search params are combined in a logical OR operation. + // Results will include rows that match **any** of the search criteria. + // + // String search params use case-insensitive matching. + // String search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions. + // example="dan%" will match "danger" and "Danzig" but not "David" + // example="D_m%" will match "Damage" and "dump" + // + // Integer search params can accept a single value or a comma separated list of values. The multiple + // values will be combined under a logical OR operation - results will match at least one of + // the given values. + // + // Most search params can accept "IS NULL" and "NOT NULL" as special expressions to match + // or exclude (respectively) rows where the column is null. + // + // Boolean search params accept only "true" and "false" as values. + // + // + rpc SearchBoards(SearchBoardsReq) returns (stream SearchBoardsStreamResponse); + + // ### Get information about a board. + // + rpc Board(BoardReq) returns (stream BoardResponse); + + // ### Update a board definition. + // + rpc UpdateBoard(UpdateBoardReq) returns (stream UpdateBoardResponse); + + // ### Delete a board. + // + rpc DeleteBoard(DeleteBoardReq) returns (stream DeleteBoardResponse); + + // ### Get information about all board items. + // + rpc AllBoardItems(AllBoardItemsReq) returns (stream AllBoardItemsStreamResponse); + + // ### Create a new board item. + // + rpc CreateBoardItem(CreateBoardItemReq) returns (stream CreateBoardItemResponse); + + // ### Get information about a board item. + // + rpc BoardItem(BoardItemReq) returns (stream BoardItemResponse); + + // ### Update a board item definition. + // + rpc UpdateBoardItem(UpdateBoardItemReq) returns (stream UpdateBoardItemResponse); + + // ### Delete a board item. + // + rpc DeleteBoardItem(DeleteBoardItemReq) returns (stream DeleteBoardItemResponse); + + // ### Get information about all board sections. + // + rpc AllBoardSections(AllBoardSectionsReq) returns (stream AllBoardSectionsStreamResponse); + + // ### Create a new board section. + // + rpc CreateBoardSection(CreateBoardSectionReq) returns (stream CreateBoardSectionResponse); + + // ### Get information about a board section. + // + rpc BoardSection(BoardSectionReq) returns (stream BoardSectionResponse); + + // ### Update a board section definition. + // + rpc UpdateBoardSection(UpdateBoardSectionReq) returns (stream UpdateBoardSectionResponse); + + // ### Delete a board section. + // + rpc DeleteBoardSection(DeleteBoardSectionReq) returns (stream DeleteBoardSectionResponse); + + + + // ColorCollection: Manage Color Collections + + // ### Get an array of all existing Color Collections + // Get a **single** color collection by id with [ColorCollection](#!/ColorCollection/color_collection) + // + // Get all **standard** color collections with [ColorCollection](#!/ColorCollection/color_collections_standard) + // + // Get all **custom** color collections with [ColorCollection](#!/ColorCollection/color_collections_custom) + // + // **Note**: Only an API user with the Admin role can call this endpoint. Unauthorized requests will return `Not Found` (404) errors. + // + // + rpc AllColorCollections(AllColorCollectionsReq) returns (stream AllColorCollectionsStreamResponse); + + // ### Create a custom color collection with the specified information + // + // Creates a new custom color collection object, returning the details, including the created id. + // + // **Update** an existing color collection with [Update Color Collection](#!/ColorCollection/update_color_collection) + // + // **Permanently delete** an existing custom color collection with [Delete Color Collection](#!/ColorCollection/delete_color_collection) + // + // **Note**: Only an API user with the Admin role can call this endpoint. Unauthorized requests will return `Not Found` (404) errors. + // + // + rpc CreateColorCollection(CreateColorCollectionReq) returns (stream CreateColorCollectionResponse); + + // ### Get an array of all existing **Custom** Color Collections + // Get a **single** color collection by id with [ColorCollection](#!/ColorCollection/color_collection) + // + // Get all **standard** color collections with [ColorCollection](#!/ColorCollection/color_collections_standard) + // + // **Note**: Only an API user with the Admin role can call this endpoint. Unauthorized requests will return `Not Found` (404) errors. + // + // + rpc ColorCollectionsCustom(ColorCollectionsCustomReq) returns (stream ColorCollectionsCustomStreamResponse); + + // ### Get an array of all existing **Standard** Color Collections + // Get a **single** color collection by id with [ColorCollection](#!/ColorCollection/color_collection) + // + // Get all **custom** color collections with [ColorCollection](#!/ColorCollection/color_collections_custom) + // + // **Note**: Only an API user with the Admin role can call this endpoint. Unauthorized requests will return `Not Found` (404) errors. + // + // + rpc ColorCollectionsStandard(ColorCollectionsStandardReq) returns (stream ColorCollectionsStandardStreamResponse); + + // ### Get the default color collection + // + // Use this to retrieve the default Color Collection. + // + // Set the default color collection with [ColorCollection](#!/ColorCollection/set_default_color_collection) + // + rpc DefaultColorCollection(DefaultColorCollectionReq) returns (stream DefaultColorCollectionResponse); + + // ### Set the global default Color Collection by ID + // + // Returns the new specified default Color Collection object. + // **Note**: Only an API user with the Admin role can call this endpoint. Unauthorized requests will return `Not Found` (404) errors. + // + // + rpc SetDefaultColorCollection(SetDefaultColorCollectionReq) returns (stream SetDefaultColorCollectionResponse); + + // ### Get a Color Collection by ID + // + // Use this to retrieve a specific Color Collection. + // Get a **single** color collection by id with [ColorCollection](#!/ColorCollection/color_collection) + // + // Get all **standard** color collections with [ColorCollection](#!/ColorCollection/color_collections_standard) + // + // Get all **custom** color collections with [ColorCollection](#!/ColorCollection/color_collections_custom) + // + // **Note**: Only an API user with the Admin role can call this endpoint. Unauthorized requests will return `Not Found` (404) errors. + // + // + rpc ColorCollection(ColorCollectionReq) returns (stream ColorCollectionResponse); + + // ### Update a custom color collection by id. + // **Note**: Only an API user with the Admin role can call this endpoint. Unauthorized requests will return `Not Found` (404) errors. + // + // + rpc UpdateColorCollection(UpdateColorCollectionReq) returns (stream UpdateColorCollectionResponse); + + // ### Delete a custom color collection by id + // + // This operation permanently deletes the identified **Custom** color collection. + // + // **Standard** color collections cannot be deleted + // + // Because multiple color collections can have the same label, they must be deleted by ID, not name. + // **Note**: Only an API user with the Admin role can call this endpoint. Unauthorized requests will return `Not Found` (404) errors. + // + // + rpc DeleteColorCollection(DeleteColorCollectionReq) returns (stream DeleteColorCollectionResponse); + + + + // Config: Manage General Configuration + + // Get the current Cloud Storage Configuration. + // + rpc CloudStorageConfiguration(CloudStorageConfigurationReq) returns (stream CloudStorageConfigurationResponse); + + // Update the current Cloud Storage Configuration. + // + rpc UpdateCloudStorageConfiguration(UpdateCloudStorageConfigurationReq) returns (stream UpdateCloudStorageConfigurationResponse); + + // ### Get the current status and content of custom welcome emails + // + rpc CustomWelcomeEmail(CustomWelcomeEmailReq) returns (stream CustomWelcomeEmailResponse); + + // Update custom welcome email setting and values. Optionally send a test email with the new content to the currently logged in user. + // + rpc UpdateCustomWelcomeEmail(UpdateCustomWelcomeEmailReq) returns (stream UpdateCustomWelcomeEmailResponse); + + // Requests to this endpoint will send a welcome email with the custom content provided in the body to the currently logged in user. + // + rpc UpdateCustomWelcomeEmailTest(UpdateCustomWelcomeEmailTestReq) returns (stream UpdateCustomWelcomeEmailTestResponse); + + // ### Retrieve the value for whether or not digest emails is enabled + // + rpc DigestEmailsEnabled(DigestEmailsEnabledReq) returns (stream DigestEmailsEnabledResponse); + + // ### Update the setting for enabling/disabling digest emails + // + rpc UpdateDigestEmailsEnabled(UpdateDigestEmailsEnabledReq) returns (stream UpdateDigestEmailsEnabledResponse); + + // ### Trigger the generation of digest email records and send them to Looker's internal system. This does not send + // any actual emails, it generates records containing content which may be of interest for users who have become inactive. + // Emails will be sent at a later time from Looker's internal system if the Digest Emails feature is enabled in settings. + rpc CreateDigestEmailSend(CreateDigestEmailSendReq) returns (stream CreateDigestEmailSendResponse); + + // ### Get Egress IP Addresses + // + // Returns the list of public egress IP Addresses for a hosted customer's instance + // + rpc PublicEgressIpAddresses(PublicEgressIpAddressesReq) returns (stream PublicEgressIpAddressesResponse); + + // ### Set the menu item name and content for internal help resources + // + rpc InternalHelpResourcesContent(InternalHelpResourcesContentReq) returns (stream InternalHelpResourcesContentResponse); + + // Update internal help resources content + // + rpc UpdateInternalHelpResourcesContent(UpdateInternalHelpResourcesContentReq) returns (stream UpdateInternalHelpResourcesContentResponse); + + // ### Get and set the options for internal help resources + // + rpc InternalHelpResources(InternalHelpResourcesReq) returns (stream InternalHelpResourcesResponse); + + // Update internal help resources settings + // + rpc UpdateInternalHelpResources(UpdateInternalHelpResourcesReq) returns (stream UpdateInternalHelpResourcesResponse); + + // ### Get all legacy features. + // + rpc AllLegacyFeatures(AllLegacyFeaturesReq) returns (stream AllLegacyFeaturesStreamResponse); + + // ### Get information about the legacy feature with a specific id. + // + rpc LegacyFeature(LegacyFeatureReq) returns (stream LegacyFeatureResponse); + + // ### Update information about the legacy feature with a specific id. + // + rpc UpdateLegacyFeature(UpdateLegacyFeatureReq) returns (stream UpdateLegacyFeatureResponse); + + // ### Get a list of locales that Looker supports. + // + rpc AllLocales(AllLocalesReq) returns (stream AllLocalesStreamResponse); + + // ### Get all mobile settings. + // + rpc MobileSettings(MobileSettingsReq) returns (stream MobileSettingsResponse); + + // ### Get Looker Settings + // + // Available settings are: + // - extension_framework_enabled + // - marketplace_auto_install_enabled + // - marketplace_enabled + // - privatelabel_configuration + // - custom_welcome_email + // - onboarding_enabled + // + // + rpc GetSetting(GetSettingReq) returns (stream GetSettingResponse); + + // ### Configure Looker Settings + // + // Available settings are: + // - extension_framework_enabled + // - marketplace_auto_install_enabled + // - marketplace_enabled + // - privatelabel_configuration + // - custom_welcome_email + // - onboarding_enabled + // + // See the `Setting` type for more information on the specific values that can be configured. + // + rpc SetSetting(SetSettingReq) returns (stream SetSettingResponse); + + // ### Configure SMTP Settings + // This API allows users to configure the SMTP settings on the Looker instance. + // This API is only supported in the OEM jar. Additionally, only admin users are authorised to call this API. + // + rpc SetSmtpSettings(SetSmtpSettingsReq) returns (stream SetSmtpSettingsResponse); + + // ### Get current SMTP status. + // + rpc SmtpStatus(SmtpStatusReq) returns (stream SmtpStatusResponse); + + // ### Get a list of timezones that Looker supports (e.g. useful for scheduling tasks). + // + rpc AllTimezones(AllTimezonesReq) returns (stream AllTimezonesStreamResponse); + + // ### Get information about all API versions supported by this Looker instance. + // + rpc Versions(VersionsReq) returns (stream VersionsResponse); + + // ### Get an API specification for this Looker instance. + // + // The specification is returned as a JSON document in Swagger 2.x format + // + rpc ApiSpec(ApiSpecReq) returns (stream ApiSpecResponse); + + // ### This feature is enabled only by special license. + // ### Gets the whitelabel configuration, which includes hiding documentation links, custom favicon uploading, etc. + // + rpc WhitelabelConfiguration(WhitelabelConfigurationReq) returns (stream WhitelabelConfigurationResponse); + + // ### Update the whitelabel configuration + // + rpc UpdateWhitelabelConfiguration(UpdateWhitelabelConfigurationReq) returns (stream UpdateWhitelabelConfigurationResponse); + + + + // Connection: Manage Database Connections + + // ### Get information about all connections. + // + rpc AllConnections(AllConnectionsReq) returns (stream AllConnectionsStreamResponse); + + // ### Create a connection using the specified configuration. + // + rpc CreateConnection(CreateConnectionReq) returns (stream CreateConnectionResponse); + + // ### Get information about a connection. + // + rpc Connection(ConnectionReq) returns (stream ConnectionResponse); + + // ### Update a connection using the specified configuration. + // + rpc UpdateConnection(UpdateConnectionReq) returns (stream UpdateConnectionResponse); + + // ### Delete a connection. + // + rpc DeleteConnection(DeleteConnectionReq) returns (stream DeleteConnectionResponse); + + // ### Delete a connection override. + // + rpc DeleteConnectionOverride(DeleteConnectionOverrideReq) returns (stream DeleteConnectionOverrideResponse); + + // ### Test an existing connection. + // + // Note that a connection's 'dialect' property has a 'connection_tests' property that lists the + // specific types of tests that the connection supports. + // + // This API is rate limited. + // + // Unsupported tests in the request will be ignored. + // + rpc TestConnection(TestConnectionReq) returns (stream TestConnectionStreamResponse); + + // ### Test a connection configuration. + // + // Note that a connection's 'dialect' property has a 'connection_tests' property that lists the + // specific types of tests that the connection supports. + // + // This API is rate limited. + // + // Unsupported tests in the request will be ignored. + // + rpc TestConnectionConfig(TestConnectionConfigReq) returns (stream TestConnectionConfigStreamResponse); + + // ### Get information about all dialects. + // + rpc AllDialectInfos(AllDialectInfosReq) returns (stream AllDialectInfosStreamResponse); + + // ### Get all External OAuth Applications. + // + // This is an OAuth Application which Looker uses to access external systems. + // + rpc AllExternalOauthApplications(AllExternalOauthApplicationsReq) returns (stream AllExternalOauthApplicationsStreamResponse); + + // ### Create an OAuth Application using the specified configuration. + // + // This is an OAuth Application which Looker uses to access external systems. + // + rpc CreateExternalOauthApplication(CreateExternalOauthApplicationReq) returns (stream CreateExternalOauthApplicationResponse); + + // ### Create OAuth User state. + // + rpc CreateOauthApplicationUserState(CreateOauthApplicationUserStateReq) returns (stream CreateOauthApplicationUserStateResponse); + + // ### Get information about all SSH Servers. + // + rpc AllSshServers(AllSshServersReq) returns (stream AllSshServersStreamResponse); + + // ### Create an SSH Server. + // + rpc CreateSshServer(CreateSshServerReq) returns (stream CreateSshServerResponse); + + // ### Get information about an SSH Server. + // + rpc SshServer(SshServerReq) returns (stream SshServerResponse); + + // ### Update an SSH Server. + // + rpc UpdateSshServer(UpdateSshServerReq) returns (stream UpdateSshServerResponse); + + // ### Delete an SSH Server. + // + rpc DeleteSshServer(DeleteSshServerReq) returns (stream DeleteSshServerResponse); + + // ### Test the SSH Server + // + rpc TestSshServer(TestSshServerReq) returns (stream TestSshServerResponse); + + // ### Get information about all SSH Tunnels. + // + rpc AllSshTunnels(AllSshTunnelsReq) returns (stream AllSshTunnelsStreamResponse); + + // ### Create an SSH Tunnel + // + rpc CreateSshTunnel(CreateSshTunnelReq) returns (stream CreateSshTunnelResponse); + + // ### Get information about an SSH Tunnel. + // + rpc SshTunnel(SshTunnelReq) returns (stream SshTunnelResponse); + + // ### Update an SSH Tunnel + // + rpc UpdateSshTunnel(UpdateSshTunnelReq) returns (stream UpdateSshTunnelResponse); + + // ### Delete an SSH Tunnel + // + rpc DeleteSshTunnel(DeleteSshTunnelReq) returns (stream DeleteSshTunnelResponse); + + // ### Test the SSH Tunnel + // + rpc TestSshTunnel(TestSshTunnelReq) returns (stream TestSshTunnelResponse); + + // ### Get the SSH public key + // + // Get the public key created for this instance to identify itself to a remote SSH server. + // + rpc SshPublicKey(SshPublicKeyReq) returns (stream SshPublicKeyResponse); + + + + // Content: Manage Content + + // ### Search Favorite Content + // + // If multiple search params are given and `filter_or` is FALSE or not specified, + // search params are combined in a logical AND operation. + // Only rows that match *all* search param criteria will be returned. + // + // If `filter_or` is TRUE, multiple search params are combined in a logical OR operation. + // Results will include rows that match **any** of the search criteria. + // + // String search params use case-insensitive matching. + // String search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions. + // example="dan%" will match "danger" and "Danzig" but not "David" + // example="D_m%" will match "Damage" and "dump" + // + // Integer search params can accept a single value or a comma separated list of values. The multiple + // values will be combined under a logical OR operation - results will match at least one of + // the given values. + // + // Most search params can accept "IS NULL" and "NOT NULL" as special expressions to match + // or exclude (respectively) rows where the column is null. + // + // Boolean search params accept only "true" and "false" as values. + // + // + rpc SearchContentFavorites(SearchContentFavoritesReq) returns (stream SearchContentFavoritesStreamResponse); + + // ### Get favorite content by its id + rpc ContentFavorite(ContentFavoriteReq) returns (stream ContentFavoriteResponse); + + // ### Delete favorite content + rpc DeleteContentFavorite(DeleteContentFavoriteReq) returns (stream DeleteContentFavoriteResponse); + + // ### Create favorite content + rpc CreateContentFavorite(CreateContentFavoriteReq) returns (stream CreateContentFavoriteResponse); + + // ### Get information about all content metadata in a space. + // + rpc AllContentMetadatas(AllContentMetadatasReq) returns (stream AllContentMetadatasStreamResponse); + + // ### Get information about an individual content metadata record. + // + rpc ContentMetadata(ContentMetadataReq) returns (stream ContentMetadataResponse); + + // ### Move a piece of content. + // + rpc UpdateContentMetadata(UpdateContentMetadataReq) returns (stream UpdateContentMetadataResponse); + + // ### All content metadata access records for a content metadata item. + // + rpc AllContentMetadataAccesses(AllContentMetadataAccessesReq) returns (stream AllContentMetadataAccessesStreamResponse); + + // ### Create content metadata access. + // + rpc CreateContentMetadataAccess(CreateContentMetadataAccessReq) returns (stream CreateContentMetadataAccessResponse); + + // ### Update type of access for content metadata. + // + rpc UpdateContentMetadataAccess(UpdateContentMetadataAccessReq) returns (stream UpdateContentMetadataAccessResponse); + + // ### Remove content metadata access. + // + rpc DeleteContentMetadataAccess(DeleteContentMetadataAccessReq) returns (stream DeleteContentMetadataAccessResponse); + + // ### Get an image representing the contents of a dashboard or look. + // + // The returned thumbnail is an abstract representation of the contents of a dashbord or look and does not + // reflect the actual data displayed in the respective visualizations. + // + rpc ContentThumbnail(ContentThumbnailReq) returns (stream ContentThumbnailResponse); + + // ### Validate All Content + // + // Performs validation of all looks and dashboards + // Returns a list of errors found as well as metadata about the content validation run. + // + rpc ContentValidation(ContentValidationReq) returns (stream ContentValidationResponse); + + // ### Search Content Views + // + // If multiple search params are given and `filter_or` is FALSE or not specified, + // search params are combined in a logical AND operation. + // Only rows that match *all* search param criteria will be returned. + // + // If `filter_or` is TRUE, multiple search params are combined in a logical OR operation. + // Results will include rows that match **any** of the search criteria. + // + // String search params use case-insensitive matching. + // String search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions. + // example="dan%" will match "danger" and "Danzig" but not "David" + // example="D_m%" will match "Damage" and "dump" + // + // Integer search params can accept a single value or a comma separated list of values. The multiple + // values will be combined under a logical OR operation - results will match at least one of + // the given values. + // + // Most search params can accept "IS NULL" and "NOT NULL" as special expressions to match + // or exclude (respectively) rows where the column is null. + // + // Boolean search params accept only "true" and "false" as values. + // + // + rpc SearchContentViews(SearchContentViewsReq) returns (stream SearchContentViewsStreamResponse); + + // ### Get a vector image representing the contents of a dashboard or look. + // + // # DEPRECATED: Use [content_thumbnail()](#!/Content/content_thumbnail) + // + // The returned thumbnail is an abstract representation of the contents of a dashbord or look and does not + // reflect the actual data displayed in the respective visualizations. + // + rpc VectorThumbnail(VectorThumbnailReq) returns (stream VectorThumbnailResponse); + + + + // Dashboard: Manage Dashboards + + // ### Get information about all active dashboards. + // + // Returns an array of **abbreviated dashboard objects**. Dashboards marked as deleted are excluded from this list. + // + // Get the **full details** of a specific dashboard by id with [dashboard()](#!/Dashboard/dashboard) + // + // Find **deleted dashboards** with [search_dashboards()](#!/Dashboard/search_dashboards) + // + rpc AllDashboards(AllDashboardsReq) returns (stream AllDashboardsStreamResponse); + + // ### Create a new dashboard + // + // Creates a new dashboard object and returns the details of the newly created dashboard. + // + // `Title` and `space_id` are required fields. + // `Space_id` must contain the id of an existing space. + // A dashboard's `title` must be unique within the space in which it resides. + // + // If you receive a 422 error response when creating a dashboard, be sure to look at the + // response body for information about exactly which fields are missing or contain invalid data. + // + // You can **update** an existing dashboard with [update_dashboard()](#!/Dashboard/update_dashboard) + // + // You can **permanently delete** an existing dashboard with [delete_dashboard()](#!/Dashboard/delete_dashboard) + // + rpc CreateDashboard(CreateDashboardReq) returns (stream CreateDashboardResponse); + + // ### Search Dashboards + // + // Returns an **array of dashboard objects** that match the specified search criteria. + // + // If multiple search params are given and `filter_or` is FALSE or not specified, + // search params are combined in a logical AND operation. + // Only rows that match *all* search param criteria will be returned. + // + // If `filter_or` is TRUE, multiple search params are combined in a logical OR operation. + // Results will include rows that match **any** of the search criteria. + // + // String search params use case-insensitive matching. + // String search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions. + // example="dan%" will match "danger" and "Danzig" but not "David" + // example="D_m%" will match "Damage" and "dump" + // + // Integer search params can accept a single value or a comma separated list of values. The multiple + // values will be combined under a logical OR operation - results will match at least one of + // the given values. + // + // Most search params can accept "IS NULL" and "NOT NULL" as special expressions to match + // or exclude (respectively) rows where the column is null. + // + // Boolean search params accept only "true" and "false" as values. + // + // + // The parameters `limit`, and `offset` are recommended for fetching results in page-size chunks. + // + // Get a **single dashboard** by id with [dashboard()](#!/Dashboard/dashboard) + // + rpc SearchDashboards(SearchDashboardsReq) returns (stream SearchDashboardsStreamResponse); + + // ### Import a LookML dashboard to a space as a UDD + // Creates a UDD (a dashboard which exists in the Looker database rather than as a LookML file) from the LookML dashboard + // and places it in the space specified. The created UDD will have a lookml_link_id which links to the original LookML dashboard. + // + // To give the imported dashboard specify a (e.g. title: "my title") in the body of your request, otherwise the imported + // dashboard will have the same title as the original LookML dashboard. + // + // For this operation to succeed the user must have permission to see the LookML dashboard in question, and have permission to + // create content in the space the dashboard is being imported to. + // + // **Sync** a linked UDD with [sync_lookml_dashboard()](#!/Dashboard/sync_lookml_dashboard) + // **Unlink** a linked UDD by setting lookml_link_id to null with [update_dashboard()](#!/Dashboard/update_dashboard) + // + rpc ImportLookmlDashboard(ImportLookmlDashboardReq) returns (stream ImportLookmlDashboardResponse); + + // ### Update all linked dashboards to match the specified LookML dashboard. + // + // Any UDD (a dashboard which exists in the Looker database rather than as a LookML file) which has a `lookml_link_id` + // property value referring to a LookML dashboard's id (model::dashboardname) will be updated so that it matches the current state of the LookML dashboard. + // + // For this operation to succeed the user must have permission to view the LookML dashboard, and only linked dashboards + // that the user has permission to update will be synced. + // + // To **link** or **unlink** a UDD set the `lookml_link_id` property with [update_dashboard()](#!/Dashboard/update_dashboard) + // + rpc SyncLookmlDashboard(SyncLookmlDashboardReq) returns (stream SyncLookmlDashboardStreamResponse); + + // ### Get information about a dashboard + // + // Returns the full details of the identified dashboard object + // + // Get a **summary list** of all active dashboards with [all_dashboards()](#!/Dashboard/all_dashboards) + // + // You can **Search** for dashboards with [search_dashboards()](#!/Dashboard/search_dashboards) + // + rpc Dashboard(DashboardReq) returns (stream DashboardResponse); + + // ### Update a dashboard + // + // You can use this function to change the string and integer properties of + // a dashboard. Nested objects such as filters, dashboard elements, or dashboard layout components + // cannot be modified by this function - use the update functions for the respective + // nested object types (like [update_dashboard_filter()](#!/3.1/Dashboard/update_dashboard_filter) to change a filter) + // to modify nested objects referenced by a dashboard. + // + // If you receive a 422 error response when updating a dashboard, be sure to look at the + // response body for information about exactly which fields are missing or contain invalid data. + // + rpc UpdateDashboard(UpdateDashboardReq) returns (stream UpdateDashboardResponse); + + // ### Delete the dashboard with the specified id + // + // Permanently **deletes** a dashboard. (The dashboard cannot be recovered after this operation.) + // + // "Soft" delete or hide a dashboard by setting its `deleted` status to `True` with [update_dashboard()](#!/Dashboard/update_dashboard). + // + // Note: When a dashboard is deleted in the UI, it is soft deleted. Use this API call to permanently remove it, if desired. + // + rpc DeleteDashboard(DeleteDashboardReq) returns (stream DeleteDashboardResponse); + + // ### Get Aggregate Table LookML for Each Query on a Dahboard + // + // Returns a JSON object that contains the dashboard id and Aggregate Table lookml + // + // + rpc DashboardAggregateTableLookml(DashboardAggregateTableLookmlReq) returns (stream DashboardAggregateTableLookmlResponse); + + // ### Get lookml of a UDD + // + // Returns a JSON object that contains the dashboard id and the full lookml + // + // + rpc DashboardLookml(DashboardLookmlReq) returns (stream DashboardLookmlResponse); + + // ### Move an existing dashboard + // + // Moves a dashboard to a specified folder, and returns the moved dashboard. + // + // `dashboard_id` and `folder_id` are required. + // `dashboard_id` and `folder_id` must already exist, and `folder_id` must be different from the current `folder_id` of the dashboard. + // + rpc MoveDashboard(MoveDashboardReq) returns (stream MoveDashboardResponse); + + // ### Creates a new dashboard object based on LookML Dashboard YAML, and returns the details of the newly created dashboard. + // + // This is equivalent to creating a LookML Dashboard and converting to a User-defined dashboard. + // + // LookML must contain valid LookML YAML code. It's recommended to use the LookML format returned + // from [dashboard_lookml()](#!/Dashboard/dashboard_lookml) as the input LookML (newlines replaced with + // ). + // + // Note that the created dashboard is not linked to any LookML Dashboard, + // i.e. [sync_lookml_dashboard()](#!/Dashboard/sync_lookml_dashboard) will not update dashboards created by this method. + // + // + rpc CreateDashboardFromLookml(CreateDashboardFromLookmlReq) returns (stream CreateDashboardFromLookmlResponse); + + // ### Copy an existing dashboard + // + // Creates a copy of an existing dashboard, in a specified folder, and returns the copied dashboard. + // + // `dashboard_id` is required, `dashboard_id` and `folder_id` must already exist if specified. + // `folder_id` will default to the existing folder. + // + // If a dashboard with the same title already exists in the target folder, the copy will have '(copy)' + // or '(copy <# of copies>)' appended. + // + rpc CopyDashboard(CopyDashboardReq) returns (stream CopyDashboardResponse); + + // ### Search Dashboard Elements + // + // Returns an **array of DashboardElement objects** that match the specified search criteria. + // + // If multiple search params are given and `filter_or` is FALSE or not specified, + // search params are combined in a logical AND operation. + // Only rows that match *all* search param criteria will be returned. + // + // If `filter_or` is TRUE, multiple search params are combined in a logical OR operation. + // Results will include rows that match **any** of the search criteria. + // + // String search params use case-insensitive matching. + // String search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions. + // example="dan%" will match "danger" and "Danzig" but not "David" + // example="D_m%" will match "Damage" and "dump" + // + // Integer search params can accept a single value or a comma separated list of values. The multiple + // values will be combined under a logical OR operation - results will match at least one of + // the given values. + // + // Most search params can accept "IS NULL" and "NOT NULL" as special expressions to match + // or exclude (respectively) rows where the column is null. + // + // Boolean search params accept only "true" and "false" as values. + // + // + rpc SearchDashboardElements(SearchDashboardElementsReq) returns (stream SearchDashboardElementsStreamResponse); + + // ### Get information about the dashboard element with a specific id. + rpc DashboardElement(DashboardElementReq) returns (stream DashboardElementResponse); + + // ### Update the dashboard element with a specific id. + rpc UpdateDashboardElement(UpdateDashboardElementReq) returns (stream UpdateDashboardElementResponse); + + // ### Delete a dashboard element with a specific id. + rpc DeleteDashboardElement(DeleteDashboardElementReq) returns (stream DeleteDashboardElementResponse); + + // ### Get information about all the dashboard elements on a dashboard with a specific id. + rpc DashboardDashboardElements(DashboardDashboardElementsReq) returns (stream DashboardDashboardElementsStreamResponse); + + // ### Create a dashboard element on the dashboard with a specific id. + rpc CreateDashboardElement(CreateDashboardElementReq) returns (stream CreateDashboardElementResponse); + + // ### Get information about the dashboard filters with a specific id. + rpc DashboardFilter(DashboardFilterReq) returns (stream DashboardFilterResponse); + + // ### Update the dashboard filter with a specific id. + rpc UpdateDashboardFilter(UpdateDashboardFilterReq) returns (stream UpdateDashboardFilterResponse); + + // ### Delete a dashboard filter with a specific id. + rpc DeleteDashboardFilter(DeleteDashboardFilterReq) returns (stream DeleteDashboardFilterResponse); + + // ### Get information about all the dashboard filters on a dashboard with a specific id. + rpc DashboardDashboardFilters(DashboardDashboardFiltersReq) returns (stream DashboardDashboardFiltersStreamResponse); + + // ### Create a dashboard filter on the dashboard with a specific id. + rpc CreateDashboardFilter(CreateDashboardFilterReq) returns (stream CreateDashboardFilterResponse); + + // ### Get information about the dashboard elements with a specific id. + rpc DashboardLayoutComponent(DashboardLayoutComponentReq) returns (stream DashboardLayoutComponentResponse); + + // ### Update the dashboard element with a specific id. + rpc UpdateDashboardLayoutComponent(UpdateDashboardLayoutComponentReq) returns (stream UpdateDashboardLayoutComponentResponse); + + // ### Get information about all the dashboard layout components for a dashboard layout with a specific id. + rpc DashboardLayoutDashboardLayoutComponents(DashboardLayoutDashboardLayoutComponentsReq) returns (stream DashboardLayoutDashboardLayoutComponentsStreamResponse); + + // ### Get information about the dashboard layouts with a specific id. + rpc DashboardLayout(DashboardLayoutReq) returns (stream DashboardLayoutResponse); + + // ### Update the dashboard layout with a specific id. + rpc UpdateDashboardLayout(UpdateDashboardLayoutReq) returns (stream UpdateDashboardLayoutResponse); + + // ### Delete a dashboard layout with a specific id. + rpc DeleteDashboardLayout(DeleteDashboardLayoutReq) returns (stream DeleteDashboardLayoutResponse); + + // ### Get information about all the dashboard elements on a dashboard with a specific id. + rpc DashboardDashboardLayouts(DashboardDashboardLayoutsReq) returns (stream DashboardDashboardLayoutsStreamResponse); + + // ### Create a dashboard layout on the dashboard with a specific id. + rpc CreateDashboardLayout(CreateDashboardLayoutReq) returns (stream CreateDashboardLayoutResponse); + + + + // DataAction: Run Data Actions + + // Perform a data action. The data action object can be obtained from query results, and used to perform an arbitrary action. + rpc PerformDataAction(PerformDataActionReq) returns (stream PerformDataActionResponse); + + // For some data actions, the remote server may supply a form requesting further user input. This endpoint takes a data action, asks the remote server to generate a form for it, and returns that form to you for presentation to the user. + rpc FetchRemoteDataActionForm(FetchRemoteDataActionFormReq) returns (stream FetchRemoteDataActionFormResponse); + + + + // Datagroup: Manage Datagroups + + // ### Get information about all datagroups. + // + rpc AllDatagroups(AllDatagroupsReq) returns (stream AllDatagroupsStreamResponse); + + // ### Get information about a datagroup. + // + rpc Datagroup(DatagroupReq) returns (stream DatagroupResponse); + + // ### Update a datagroup using the specified params. + // + rpc UpdateDatagroup(UpdateDatagroupReq) returns (stream UpdateDatagroupResponse); + + + + // DerivedTable: View Derived Table graphs + + // ### Discover information about derived tables + // + rpc GraphDerivedTablesForModel(GraphDerivedTablesForModelReq) returns (stream GraphDerivedTablesForModelResponse); + + // ### Get the subgraph representing this derived table and its dependencies. + // + rpc GraphDerivedTablesForView(GraphDerivedTablesForViewReq) returns (stream GraphDerivedTablesForViewResponse); + + // Enqueue materialization for a PDT with the given model name and view name + rpc StartPdtBuild(StartPdtBuildReq) returns (stream StartPdtBuildResponse); + + // Check status of PDT materialization + rpc CheckPdtBuild(CheckPdtBuildReq) returns (stream CheckPdtBuildResponse); + + // Stop a PDT materialization + rpc StopPdtBuild(StopPdtBuildReq) returns (stream StopPdtBuildResponse); + + + + // Folder: Manage Folders + + // Search for folders by creator id, parent id, name, etc + rpc SearchFolders(SearchFoldersReq) returns (stream SearchFoldersStreamResponse); + + // ### Get information about the folder with a specific id. + rpc Folder(FolderReq) returns (stream FolderResponse); + + // ### Update the folder with a specific id. + rpc UpdateFolder(UpdateFolderReq) returns (stream UpdateFolderResponse); + + // ### Delete the folder with a specific id including any children folders. + // **DANGER** this will delete all looks and dashboards in the folder. + // + rpc DeleteFolder(DeleteFolderReq) returns (stream DeleteFolderResponse); + + // ### Get information about all folders. + // + // In API 3.x, this will not return empty personal folders, unless they belong to the calling user, + // or if they contain soft-deleted content. + // + // In API 4.0+, all personal folders will be returned. + // + // + rpc AllFolders(AllFoldersReq) returns (stream AllFoldersStreamResponse); + + // ### Create a folder with specified information. + // + // Caller must have permission to edit the parent folder and to create folders, otherwise the request + // returns 404 Not Found. + // + rpc CreateFolder(CreateFolderReq) returns (stream CreateFolderResponse); + + // ### Get the children of a folder. + rpc FolderChildren(FolderChildrenReq) returns (stream FolderChildrenStreamResponse); + + // ### Search the children of a folder + rpc FolderChildrenSearch(FolderChildrenSearchReq) returns (stream FolderChildrenSearchStreamResponse); + + // ### Get the parent of a folder + rpc FolderParent(FolderParentReq) returns (stream FolderParentResponse); + + // ### Get the ancestors of a folder + rpc FolderAncestors(FolderAncestorsReq) returns (stream FolderAncestorsStreamResponse); + + // ### Get all looks in a folder. + // In API 3.x, this will return all looks in a folder, including looks in the trash. + // In API 4.0+, all looks in a folder will be returned, excluding looks in the trash. + // + rpc FolderLooks(FolderLooksReq) returns (stream FolderLooksStreamResponse); + + // ### Get the dashboards in a folder + rpc FolderDashboards(FolderDashboardsReq) returns (stream FolderDashboardsStreamResponse); + + + + // Group: Manage Groups + + // ### Get information about all groups. + // + rpc AllGroups(AllGroupsReq) returns (stream AllGroupsStreamResponse); + + // ### Creates a new group (admin only). + // + rpc CreateGroup(CreateGroupReq) returns (stream CreateGroupResponse); + + // ### Search groups + // + // Returns all group records that match the given search criteria. + // + // If multiple search params are given and `filter_or` is FALSE or not specified, + // search params are combined in a logical AND operation. + // Only rows that match *all* search param criteria will be returned. + // + // If `filter_or` is TRUE, multiple search params are combined in a logical OR operation. + // Results will include rows that match **any** of the search criteria. + // + // String search params use case-insensitive matching. + // String search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions. + // example="dan%" will match "danger" and "Danzig" but not "David" + // example="D_m%" will match "Damage" and "dump" + // + // Integer search params can accept a single value or a comma separated list of values. The multiple + // values will be combined under a logical OR operation - results will match at least one of + // the given values. + // + // Most search params can accept "IS NULL" and "NOT NULL" as special expressions to match + // or exclude (respectively) rows where the column is null. + // + // Boolean search params accept only "true" and "false" as values. + // + // + rpc SearchGroups(SearchGroupsReq) returns (stream SearchGroupsStreamResponse); + + // ### Search groups include roles + // + // Returns all group records that match the given search criteria, and attaches any associated roles. + // + // If multiple search params are given and `filter_or` is FALSE or not specified, + // search params are combined in a logical AND operation. + // Only rows that match *all* search param criteria will be returned. + // + // If `filter_or` is TRUE, multiple search params are combined in a logical OR operation. + // Results will include rows that match **any** of the search criteria. + // + // String search params use case-insensitive matching. + // String search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions. + // example="dan%" will match "danger" and "Danzig" but not "David" + // example="D_m%" will match "Damage" and "dump" + // + // Integer search params can accept a single value or a comma separated list of values. The multiple + // values will be combined under a logical OR operation - results will match at least one of + // the given values. + // + // Most search params can accept "IS NULL" and "NOT NULL" as special expressions to match + // or exclude (respectively) rows where the column is null. + // + // Boolean search params accept only "true" and "false" as values. + // + // + rpc SearchGroupsWithRoles(SearchGroupsWithRolesReq) returns (stream SearchGroupsWithRolesStreamResponse); + + // ### Search groups include hierarchy + // + // Returns all group records that match the given search criteria, and attaches + // associated role_ids and parent group_ids. + // + // If multiple search params are given and `filter_or` is FALSE or not specified, + // search params are combined in a logical AND operation. + // Only rows that match *all* search param criteria will be returned. + // + // If `filter_or` is TRUE, multiple search params are combined in a logical OR operation. + // Results will include rows that match **any** of the search criteria. + // + // String search params use case-insensitive matching. + // String search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions. + // example="dan%" will match "danger" and "Danzig" but not "David" + // example="D_m%" will match "Damage" and "dump" + // + // Integer search params can accept a single value or a comma separated list of values. The multiple + // values will be combined under a logical OR operation - results will match at least one of + // the given values. + // + // Most search params can accept "IS NULL" and "NOT NULL" as special expressions to match + // or exclude (respectively) rows where the column is null. + // + // Boolean search params accept only "true" and "false" as values. + // + // + rpc SearchGroupsWithHierarchy(SearchGroupsWithHierarchyReq) returns (stream SearchGroupsWithHierarchyStreamResponse); + + // ### Get information about a group. + // + rpc Group(GroupReq) returns (stream GroupResponse); + + // ### Updates the a group (admin only). + rpc UpdateGroup(UpdateGroupReq) returns (stream UpdateGroupResponse); + + // ### Deletes a group (admin only). + // + rpc DeleteGroup(DeleteGroupReq) returns (stream DeleteGroupResponse); + + // ### Get information about all the groups in a group + // + rpc AllGroupGroups(AllGroupGroupsReq) returns (stream AllGroupGroupsStreamResponse); + + // ### Adds a new group to a group. + // + rpc AddGroupGroup(AddGroupGroupReq) returns (stream AddGroupGroupResponse); + + // ### Get information about all the users directly included in a group. + // + rpc AllGroupUsers(AllGroupUsersReq) returns (stream AllGroupUsersStreamResponse); + + // ### Adds a new user to a group. + // + rpc AddGroupUser(AddGroupUserReq) returns (stream AddGroupUserResponse); + + // ### Removes a user from a group. + // + rpc DeleteGroupUser(DeleteGroupUserReq) returns (stream DeleteGroupUserResponse); + + // ### Removes a group from a group. + // + rpc DeleteGroupFromGroup(DeleteGroupFromGroupReq) returns (stream DeleteGroupFromGroupResponse); + + // ### Set the value of a user attribute for a group. + // + // For information about how user attribute values are calculated, see [Set User Attribute Group Values](#!/UserAttribute/set_user_attribute_group_values). + // + rpc UpdateUserAttributeGroupValue(UpdateUserAttributeGroupValueReq) returns (stream UpdateUserAttributeGroupValueResponse); + + // ### Remove a user attribute value from a group. + // + rpc DeleteUserAttributeGroupValue(DeleteUserAttributeGroupValueReq) returns (stream DeleteUserAttributeGroupValueResponse); + + + + // Homepage: Manage Homepage + + // ### Get information about the primary homepage's sections. + // + rpc AllPrimaryHomepageSections(AllPrimaryHomepageSectionsReq) returns (stream AllPrimaryHomepageSectionsStreamResponse); + + + + // Integration: Manage Integrations + + // ### Get information about all Integration Hubs. + // + rpc AllIntegrationHubs(AllIntegrationHubsReq) returns (stream AllIntegrationHubsStreamResponse); + + // ### Create a new Integration Hub. + // + // This API is rate limited to prevent it from being used for SSRF attacks + // + rpc CreateIntegrationHub(CreateIntegrationHubReq) returns (stream CreateIntegrationHubResponse); + + // ### Get information about a Integration Hub. + // + rpc IntegrationHub(IntegrationHubReq) returns (stream IntegrationHubResponse); + + // ### Update a Integration Hub definition. + // + // This API is rate limited to prevent it from being used for SSRF attacks + // + rpc UpdateIntegrationHub(UpdateIntegrationHubReq) returns (stream UpdateIntegrationHubResponse); + + // ### Delete a Integration Hub. + // + rpc DeleteIntegrationHub(DeleteIntegrationHubReq) returns (stream DeleteIntegrationHubResponse); + + // Accepts the legal agreement for a given integration hub. This only works for integration hubs that have legal_agreement_required set to true and legal_agreement_signed set to false. + rpc AcceptIntegrationHubLegalAgreement(AcceptIntegrationHubLegalAgreementReq) returns (stream AcceptIntegrationHubLegalAgreementResponse); + + // ### Get information about all Integrations. + // + rpc AllIntegrations(AllIntegrationsReq) returns (stream AllIntegrationsStreamResponse); + + // ### Get information about a Integration. + // + rpc Integration(IntegrationReq) returns (stream IntegrationResponse); + + // ### Update parameters on a Integration. + // + rpc UpdateIntegration(UpdateIntegrationReq) returns (stream UpdateIntegrationResponse); + + // Returns the Integration form for presentation to the user. + rpc FetchIntegrationForm(FetchIntegrationFormReq) returns (stream FetchIntegrationFormResponse); + + // Tests the integration to make sure all the settings are working. + rpc TestIntegration(TestIntegrationReq) returns (stream TestIntegrationResponse); + + + + // Look: Run and Manage Looks + + // ### Get information about all active Looks + // + // Returns an array of **abbreviated Look objects** describing all the looks that the caller has access to. Soft-deleted Looks are **not** included. + // + // Get the **full details** of a specific look by id with [look(id)](#!/Look/look) + // + // Find **soft-deleted looks** with [search_looks()](#!/Look/search_looks) + // + rpc AllLooks(AllLooksReq) returns (stream AllLooksStreamResponse); + + // ### Create a Look + // + // To create a look to display query data, first create the query with [create_query()](#!/Query/create_query) + // then assign the query's id to the `query_id` property in the call to `create_look()`. + // + // To place the look into a particular space, assign the space's id to the `space_id` property + // in the call to `create_look()`. + // + rpc CreateLook(CreateLookReq) returns (stream CreateLookResponse); + + // ### Search Looks + // + // Returns an **array of Look objects** that match the specified search criteria. + // + // If multiple search params are given and `filter_or` is FALSE or not specified, + // search params are combined in a logical AND operation. + // Only rows that match *all* search param criteria will be returned. + // + // If `filter_or` is TRUE, multiple search params are combined in a logical OR operation. + // Results will include rows that match **any** of the search criteria. + // + // String search params use case-insensitive matching. + // String search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions. + // example="dan%" will match "danger" and "Danzig" but not "David" + // example="D_m%" will match "Damage" and "dump" + // + // Integer search params can accept a single value or a comma separated list of values. The multiple + // values will be combined under a logical OR operation - results will match at least one of + // the given values. + // + // Most search params can accept "IS NULL" and "NOT NULL" as special expressions to match + // or exclude (respectively) rows where the column is null. + // + // Boolean search params accept only "true" and "false" as values. + // + // + // Get a **single look** by id with [look(id)](#!/Look/look) + // + rpc SearchLooks(SearchLooksReq) returns (stream SearchLooksStreamResponse); + + // ### Get a Look. + // + // Returns detailed information about a Look and its associated Query. + // + // + rpc Look(LookReq) returns (stream LookResponse); + + // ### Modify a Look + // + // Use this function to modify parts of a look. Property values given in a call to `update_look` are + // applied to the existing look, so there's no need to include properties whose values are not changing. + // It's best to specify only the properties you want to change and leave everything else out + // of your `update_look` call. **Look properties marked 'read-only' will be ignored.** + // + // When a user deletes a look in the Looker UI, the look data remains in the database but is + // marked with a deleted flag ("soft-deleted"). Soft-deleted looks can be undeleted (by an admin) + // if the delete was in error. + // + // To soft-delete a look via the API, use [update_look()](#!/Look/update_look) to change the look's `deleted` property to `true`. + // You can undelete a look by calling `update_look` to change the look's `deleted` property to `false`. + // + // Soft-deleted looks are excluded from the results of [all_looks()](#!/Look/all_looks) and [search_looks()](#!/Look/search_looks), so they + // essentially disappear from view even though they still reside in the db. + // In API 3.1 and later, you can pass `deleted: true` as a parameter to [search_looks()](#!/3.1/Look/search_looks) to list soft-deleted looks. + // + // NOTE: [delete_look()](#!/Look/delete_look) performs a "hard delete" - the look data is removed from the Looker + // database and destroyed. There is no "undo" for `delete_look()`. + // + rpc UpdateLook(UpdateLookReq) returns (stream UpdateLookResponse); + + // ### Permanently Delete a Look + // + // This operation **permanently** removes a look from the Looker database. + // + // NOTE: There is no "undo" for this kind of delete. + // + // For information about soft-delete (which can be undone) see [update_look()](#!/Look/update_look). + // + rpc DeleteLook(DeleteLookReq) returns (stream DeleteLookResponse); + + // ### Run a Look + // + // Runs a given look's query and returns the results in the requested format. + // + // Supported formats: + // + // | result_format | Description + // | :-----------: | :--- | + // | json | Plain json + // | json_detail | Row data plus metadata describing the fields, pivots, table calcs, and other aspects of the query + // | csv | Comma separated values with a header + // | txt | Tab separated values with a header + // | html | Simple html + // | md | Simple markdown + // | xlsx | MS Excel spreadsheet + // | sql | Returns the generated SQL rather than running the query + // | png | A PNG image of the visualization of the query + // | jpg | A JPG image of the visualization of the query + // + // + // + rpc RunLook(RunLookReq) returns (stream RunLookResponse); + + // ### Copy an existing look + // + // Creates a copy of an existing look, in a specified folder, and returns the copied look. + // + // `look_id` and `folder_id` are required. + // + // `look_id` and `folder_id` must already exist, and `folder_id` must be different from the current `folder_id` of the dashboard. + // + rpc CopyLook(CopyLookReq) returns (stream CopyLookResponse); + + // ### Move an existing look + // + // Moves a look to a specified folder, and returns the moved look. + // + // `look_id` and `folder_id` are required. + // `look_id` and `folder_id` must already exist, and `folder_id` must be different from the current `folder_id` of the dashboard. + // + rpc MoveLook(MoveLookReq) returns (stream MoveLookResponse); + + + + // LookmlModel: Manage LookML Models + + // ### Get information about all lookml models. + // + rpc AllLookmlModels(AllLookmlModelsReq) returns (stream AllLookmlModelsStreamResponse); + + // ### Create a lookml model using the specified configuration. + // + rpc CreateLookmlModel(CreateLookmlModelReq) returns (stream CreateLookmlModelResponse); + + // ### Get information about a lookml model. + // + rpc LookmlModel(LookmlModelReq) returns (stream LookmlModelResponse); + + // ### Update a lookml model using the specified configuration. + // + rpc UpdateLookmlModel(UpdateLookmlModelReq) returns (stream UpdateLookmlModelResponse); + + // ### Delete a lookml model. + // + rpc DeleteLookmlModel(DeleteLookmlModelReq) returns (stream DeleteLookmlModelResponse); + + // ### Get information about a lookml model explore. + // + rpc LookmlModelExplore(LookmlModelExploreReq) returns (stream LookmlModelExploreResponse); + + + + // Metadata: Connection Metadata Features + + // ### Field name suggestions for a model and view + // + // `filters` is a string hash of values, with the key as the field name and the string value as the filter expression: + // + // ```ruby + // {'users.age': '>=60'} + // ``` + // + // or + // + // ```ruby + // {'users.age': '<30'} + // ``` + // + // or + // + // ```ruby + // {'users.age': '=50'} + // ``` + // + rpc ModelFieldnameSuggestions(ModelFieldnameSuggestionsReq) returns (stream ModelFieldnameSuggestionsResponse); + + // ### Get a single model + // + // + rpc GetModel(GetModelReq) returns (stream GetModelResponse); + + // ### List databases available to this connection + // + // Certain dialects can support multiple databases per single connection. + // If this connection supports multiple databases, the database names will be returned in an array. + // + // Connections using dialects that do not support multiple databases will return an empty array. + // + // **Note**: [Connection Features](#!/Metadata/connection_features) can be used to determine if a connection supports + // multiple databases. + // + rpc ConnectionDatabases(ConnectionDatabasesReq) returns (stream ConnectionDatabasesStreamResponse); + + // ### Retrieve metadata features for this connection + // + // Returns a list of feature names with `true` (available) or `false` (not available) + // + // + rpc ConnectionFeatures(ConnectionFeaturesReq) returns (stream ConnectionFeaturesResponse); + + // ### Get the list of schemas and tables for a connection + // + // + rpc ConnectionSchemas(ConnectionSchemasReq) returns (stream ConnectionSchemasStreamResponse); + + // ### Get the list of tables for a schema + // + // For dialects that support multiple databases, optionally identify which to use. If not provided, the default + // database for the connection will be used. + // + // For dialects that do **not** support multiple databases, **do not use** the database parameter + // + rpc ConnectionTables(ConnectionTablesReq) returns (stream ConnectionTablesStreamResponse); + + // ### Get the columns (and therefore also the tables) in a specific schema + // + // + rpc ConnectionColumns(ConnectionColumnsReq) returns (stream ConnectionColumnsStreamResponse); + + // ### Search a connection for columns matching the specified name + // + // **Note**: `column_name` must be a valid column name. It is not a search pattern. + // + rpc ConnectionSearchColumns(ConnectionSearchColumnsReq) returns (stream ConnectionSearchColumnsStreamResponse); + + // ### Connection cost estimating + // + // Assign a `sql` statement to the body of the request. e.g., for Ruby, `{sql: 'select * from users'}` + // + // **Note**: If the connection's dialect has no support for cost estimates, an error will be returned + // + rpc ConnectionCostEstimate(ConnectionCostEstimateReq) returns (stream ConnectionCostEstimateResponse); + + + + // Project: Manage Projects + + // ### Generate Lockfile for All LookML Dependencies + // + // Git must have been configured, must be in dev mode and deploy permission required + // + // Install_all is a two step process + // 1. For each remote_dependency in a project the dependency manager will resolve any ambiguous ref. + // 2. The project will then write out a lockfile including each remote_dependency with its resolved ref. + // + // + rpc LockAll(LockAllReq) returns (stream LockAllResponse); + + // ### Get All Git Branches + // + // Returns a list of git branches in the project repository + // + rpc AllGitBranches(AllGitBranchesReq) returns (stream AllGitBranchesStreamResponse); + + // ### Get the Current Git Branch + // + // Returns the git branch currently checked out in the given project repository + // + rpc GitBranch(GitBranchReq) returns (stream GitBranchResponse); + + // ### Checkout and/or reset --hard an existing Git Branch + // + // Only allowed in development mode + // - Call `update_session` to select the 'dev' workspace. + // + // Checkout an existing branch if name field is different from the name of the currently checked out branch. + // + // Optionally specify a branch name, tag name or commit SHA to which the branch should be reset. + // **DANGER** hard reset will be force pushed to the remote. Unsaved changes and commits may be permanently lost. + // + // + rpc UpdateGitBranch(UpdateGitBranchReq) returns (stream UpdateGitBranchResponse); + + // ### Create and Checkout a Git Branch + // + // Creates and checks out a new branch in the given project repository + // Only allowed in development mode + // - Call `update_session` to select the 'dev' workspace. + // + // Optionally specify a branch name, tag name or commit SHA as the start point in the ref field. + // If no ref is specified, HEAD of the current branch will be used as the start point for the new branch. + // + // + rpc CreateGitBranch(CreateGitBranchReq) returns (stream CreateGitBranchResponse); + + // ### Get the specified Git Branch + // + // Returns the git branch specified in branch_name path param if it exists in the given project repository + // + rpc FindGitBranch(FindGitBranchReq) returns (stream FindGitBranchResponse); + + // ### Delete the specified Git Branch + // + // Delete git branch specified in branch_name path param from local and remote of specified project repository + // + rpc DeleteGitBranch(DeleteGitBranchReq) returns (stream DeleteGitBranchResponse); + + // ### Deploy a Remote Branch or Ref to Production + // + // Git must have been configured and deploy permission required. + // + // Deploy is a one/two step process + // 1. If this is the first deploy of this project, create the production project with git repository. + // 2. Pull the branch or ref into the production project. + // + // Can only specify either a branch or a ref. + // + // + rpc DeployRefToProduction(DeployRefToProductionReq) returns (stream DeployRefToProductionResponse); + + // ### Deploy LookML from this Development Mode Project to Production + // + // Git must have been configured, must be in dev mode and deploy permission required + // + // Deploy is a two / three step process: + // + // 1. Push commits in current branch of dev mode project to the production branch (origin/master). + // Note a. This step is skipped in read-only projects. + // Note b. If this step is unsuccessful for any reason (e.g. rejected non-fastforward because production branch has + // commits not in current branch), subsequent steps will be skipped. + // 2. If this is the first deploy of this project, create the production project with git repository. + // 3. Pull the production branch into the production project. + // + // + rpc DeployToProduction(DeployToProductionReq) returns (stream DeployToProductionResponse); + + // ### Reset a project to the revision of the project that is in production. + // + // **DANGER** this will delete any changes that have not been pushed to a remote repository. + // + rpc ResetProjectToProduction(ResetProjectToProductionReq) returns (stream ResetProjectToProductionResponse); + + // ### Reset a project development branch to the revision of the project that is on the remote. + // + // **DANGER** this will delete any changes that have not been pushed to a remote repository. + // + rpc ResetProjectToRemote(ResetProjectToRemoteReq) returns (stream ResetProjectToRemoteResponse); + + // ### Get All Projects + // + // Returns all projects visible to the current user + // + rpc AllProjects(AllProjectsReq) returns (stream AllProjectsStreamResponse); + + // ### Create A Project + // + // dev mode required. + // - Call `update_session` to select the 'dev' workspace. + // + // `name` is required. + // `git_remote_url` is not allowed. To configure Git for the newly created project, follow the instructions in `update_project`. + // + // + rpc CreateProject(CreateProjectReq) returns (stream CreateProjectResponse); + + // ### Get A Project + // + // Returns the project with the given project id + // + rpc Project(ProjectReq) returns (stream ProjectResponse); + + // ### Update Project Configuration + // + // Apply changes to a project's configuration. + // + // + // #### Configuring Git for a Project + // + // To set up a Looker project with a remote git repository, follow these steps: + // + // 1. Call `update_session` to select the 'dev' workspace. + // 1. Call `create_git_deploy_key` to create a new deploy key for the project + // 1. Copy the deploy key text into the remote git repository's ssh key configuration + // 1. Call `update_project` to set project's `git_remote_url` ()and `git_service_name`, if necessary). + // + // When you modify a project's `git_remote_url`, Looker connects to the remote repository to fetch + // metadata. The remote git repository MUST be configured with the Looker-generated deploy + // key for this project prior to setting the project's `git_remote_url`. + // + // To set up a Looker project with a git repository residing on the Looker server (a 'bare' git repo): + // + // 1. Call `update_session` to select the 'dev' workspace. + // 1. Call `update_project` setting `git_remote_url` to null and `git_service_name` to "bare". + // + // + rpc UpdateProject(UpdateProjectReq) returns (stream UpdateProjectResponse); + + // ### Get A Projects Manifest object + // + // Returns the project with the given project id + // + rpc Manifest(ManifestReq) returns (stream ManifestResponse); + + // ### Git Deploy Key + // + // Returns the ssh public key previously created for a project's git repository. + // + rpc GitDeployKey(GitDeployKeyReq) returns (stream GitDeployKeyResponse); + + // ### Create Git Deploy Key + // + // Create a public/private key pair for authenticating ssh git requests from Looker to a remote git repository + // for a particular Looker project. + // + // Returns the public key of the generated ssh key pair. + // + // Copy this public key to your remote git repository's ssh keys configuration so that the remote git service can + // validate and accept git requests from the Looker server. + // + rpc CreateGitDeployKey(CreateGitDeployKeyReq) returns (stream CreateGitDeployKeyResponse); + + // ### Get Cached Project Validation Results + // + // Returns the cached results of a previous project validation calculation, if any. + // Returns http status 204 No Content if no validation results exist. + // + // Validating the content of all the files in a project can be computationally intensive + // for large projects. Use this API to simply fetch the results of the most recent + // project validation rather than revalidating the entire project from scratch. + // + // A value of `"stale": true` in the response indicates that the project has changed since + // the cached validation results were computed. The cached validation results may no longer + // reflect the current state of the project. + // + rpc ProjectValidationResults(ProjectValidationResultsReq) returns (stream ProjectValidationResultsResponse); + + // ### Validate Project + // + // Performs lint validation of all lookml files in the project. + // Returns a list of errors found, if any. + // + // Validating the content of all the files in a project can be computationally intensive + // for large projects. For best performance, call `validate_project(project_id)` only + // when you really want to recompute project validation. To quickly display the results of + // the most recent project validation (without recomputing), use `project_validation_results(project_id)` + // + rpc ValidateProject(ValidateProjectReq) returns (stream ValidateProjectResponse); + + // ### Get Project Workspace + // + // Returns information about the state of the project files in the currently selected workspace + // + rpc ProjectWorkspace(ProjectWorkspaceReq) returns (stream ProjectWorkspaceResponse); + + // ### Get All Project Files + // + // Returns a list of the files in the project + // + rpc AllProjectFiles(AllProjectFilesReq) returns (stream AllProjectFilesStreamResponse); + + // ### Get Project File Info + // + // Returns information about a file in the project + // + rpc ProjectFile(ProjectFileReq) returns (stream ProjectFileResponse); + + // ### Get All Git Connection Tests + // + // dev mode required. + // - Call `update_session` to select the 'dev' workspace. + // + // Returns a list of tests which can be run against a project's (or the dependency project for the provided remote_url) git connection. Call [Run Git Connection Test](#!/Project/run_git_connection_test) to execute each test in sequence. + // + // Tests are ordered by increasing specificity. Tests should be run in the order returned because later tests require functionality tested by tests earlier in the test list. + // + // For example, a late-stage test for write access is meaningless if connecting to the git server (an early test) is failing. + // + rpc AllGitConnectionTests(AllGitConnectionTestsReq) returns (stream AllGitConnectionTestsStreamResponse); + + // ### Run a git connection test + // + // Run the named test on the git service used by this project (or the dependency project for the provided remote_url) and return the result. This + // is intended to help debug git connections when things do not work properly, to give + // more helpful information about why a git url is not working with Looker. + // + // Tests should be run in the order they are returned by [Get All Git Connection Tests](#!/Project/all_git_connection_tests). + // + rpc RunGitConnectionTest(RunGitConnectionTestReq) returns (stream RunGitConnectionTestResponse); + + // ### Get All LookML Tests + // + // Returns a list of tests which can be run to validate a project's LookML code and/or the underlying data, + // optionally filtered by the file id. + // Call [Run LookML Test](#!/Project/run_lookml_test) to execute tests. + // + rpc AllLookmlTests(AllLookmlTestsReq) returns (stream AllLookmlTestsStreamResponse); + + // ### Run LookML Tests + // + // Runs all tests in the project, optionally filtered by file, test, and/or model. + // + rpc RunLookmlTest(RunLookmlTestReq) returns (stream RunLookmlTestStreamResponse); + + // ### Creates a tag for the most recent commit, or a specific ref is a SHA is provided + // + // This is an internal-only, undocumented route. + // + rpc TagRef(TagRefReq) returns (stream TagRefResponse); + + // ### Configure Repository Credential for a remote dependency + // + // Admin required. + // + // `root_project_id` is required. + // `credential_id` is required. + // + // + rpc UpdateRepositoryCredential(UpdateRepositoryCredentialReq) returns (stream UpdateRepositoryCredentialResponse); + + // ### Repository Credential for a remote dependency + // + // Admin required. + // + // `root_project_id` is required. + // `credential_id` is required. + // + rpc DeleteRepositoryCredential(DeleteRepositoryCredentialReq) returns (stream DeleteRepositoryCredentialResponse); + + // ### Get all Repository Credentials for a project + // + // `root_project_id` is required. + // + rpc GetAllRepositoryCredentials(GetAllRepositoryCredentialsReq) returns (stream GetAllRepositoryCredentialsStreamResponse); + + + + // Query: Run and Manage Queries + + // ### Create an async query task + // + // Creates a query task (job) to run a previously created query asynchronously. Returns a Query Task ID. + // + // Use [query_task(query_task_id)](#!/Query/query_task) to check the execution status of the query task. + // After the query task status reaches "Complete", use [query_task_results(query_task_id)](#!/Query/query_task_results) to fetch the results of the query. + // + rpc CreateQueryTask(CreateQueryTaskReq) returns (stream CreateQueryTaskResponse); + + // ### Fetch results of multiple async queries + // + // Returns the results of multiple async queries in one request. + // + // For Query Tasks that are not completed, the response will include the execution status of the Query Task but will not include query results. + // Query Tasks whose results have expired will have a status of 'expired'. + // If the user making the API request does not have sufficient privileges to view a Query Task result, the result will have a status of 'missing' + // + rpc QueryTaskMultiResults(QueryTaskMultiResultsReq) returns (stream QueryTaskMultiResultsResponse); + + // ### Get Query Task details + // + // Use this function to check the status of an async query task. After the status + // reaches "Complete", you can call [query_task_results(query_task_id)](#!/Query/query_task_results) to + // retrieve the results of the query. + // + // Use [create_query_task()](#!/Query/create_query_task) to create an async query task. + // + rpc QueryTask(QueryTaskReq) returns (stream QueryTaskResponse); + + // ### Get Async Query Results + // + // Returns the results of an async query task if the query has completed. + // + // If the query task is still running or waiting to run, this function returns 204 No Content. + // + // If the query task ID is invalid or the cached results of the query task have expired, this function returns 404 Not Found. + // + // Use [query_task(query_task_id)](#!/Query/query_task) to check the execution status of the query task + // Call query_task_results only after the query task status reaches "Complete". + // + // You can also use [query_task_multi_results()](#!/Query/query_task_multi_results) retrieve the + // results of multiple async query tasks at the same time. + // + // #### SQL Error Handling: + // If the query fails due to a SQL db error, how this is communicated depends on the result_format you requested in `create_query_task()`. + // + // For `json_detail` result_format: `query_task_results()` will respond with HTTP status '200 OK' and db SQL error info + // will be in the `errors` property of the response object. The 'data' property will be empty. + // + // For all other result formats: `query_task_results()` will respond with HTTP status `400 Bad Request` and some db SQL error info + // will be in the message of the 400 error response, but not as detailed as expressed in `json_detail.errors`. + // These data formats can only carry row data, and error info is not row data. + // + rpc QueryTaskResults(QueryTaskResultsReq) returns (stream QueryTaskResultsResponse); + + // ### Get a previously created query by id. + // + // A Looker query object includes the various parameters that define a database query that has been run or + // could be run in the future. These parameters include: model, view, fields, filters, pivots, etc. + // Query *results* are not part of the query object. + // + // Query objects are unique and immutable. Query objects are created automatically in Looker as users explore data. + // Looker does not delete them; they become part of the query history. When asked to create a query for + // any given set of parameters, Looker will first try to find an existing query object with matching + // parameters and will only create a new object when an appropriate object can not be found. + // + // This 'get' method is used to get the details about a query for a given id. See the other methods here + // to 'create' and 'run' queries. + // + // Note that some fields like 'filter_config' and 'vis_config' etc are specific to how the Looker UI + // builds queries and visualizations and are not generally useful for API use. They are not required when + // creating new queries and can usually just be ignored. + // + // + rpc Query(QueryReq) returns (stream QueryResponse); + + // ### Get the query for a given query slug. + // + // This returns the query for the 'slug' in a query share URL. + // + // The 'slug' is a randomly chosen short string that is used as an alternative to the query's id value + // for use in URLs etc. This method exists as a convenience to help you use the API to 'find' queries that + // have been created using the Looker UI. + // + // You can use the Looker explore page to build a query and then choose the 'Share' option to + // show the share url for the query. Share urls generally look something like 'https://looker.yourcompany/x/vwGSbfc'. + // The trailing 'vwGSbfc' is the share slug. You can pass that string to this api method to get details about the query. + // Those details include the 'id' that you can use to run the query. Or, you can copy the query body + // (perhaps with your own modification) and use that as the basis to make/run new queries. + // + // This will also work with slugs from Looker explore urls like + // 'https://looker.yourcompany/explore/ecommerce/orders?qid=aogBgL6o3cKK1jN3RoZl5s'. In this case + // 'aogBgL6o3cKK1jN3RoZl5s' is the slug. + // + rpc QueryForSlug(QueryForSlugReq) returns (stream QueryForSlugResponse); + + // ### Create a query. + // + // This allows you to create a new query that you can later run. Looker queries are immutable once created + // and are not deleted. If you create a query that is exactly like an existing query then the existing query + // will be returned and no new query will be created. Whether a new query is created or not, you can use + // the 'id' in the returned query with the 'run' method. + // + // The query parameters are passed as json in the body of the request. + // + // + rpc CreateQuery(CreateQueryReq) returns (stream CreateQueryResponse); + + // ### Run a saved query. + // + // This runs a previously saved query. You can use this on a query that was generated in the Looker UI + // or one that you have explicitly created using the API. You can also use a query 'id' from a saved 'Look'. + // + // The 'result_format' parameter specifies the desired structure and format of the response. + // + // Supported formats: + // + // | result_format | Description + // | :-----------: | :--- | + // | json | Plain json + // | json_detail | Row data plus metadata describing the fields, pivots, table calcs, and other aspects of the query + // | csv | Comma separated values with a header + // | txt | Tab separated values with a header + // | html | Simple html + // | md | Simple markdown + // | xlsx | MS Excel spreadsheet + // | sql | Returns the generated SQL rather than running the query + // | png | A PNG image of the visualization of the query + // | jpg | A JPG image of the visualization of the query + // + // + // + rpc RunQuery(RunQueryReq) returns (stream RunQueryResponse); + + // ### Run the query that is specified inline in the posted body. + // + // This allows running a query as defined in json in the posted body. This combines + // the two actions of posting & running a query into one step. + // + // Here is an example body in json: + // ``` + // { + // "model":"thelook", + // "view":"inventory_items", + // "fields":["category.name","inventory_items.days_in_inventory_tier","products.count"], + // "filters":{"category.name":"socks"}, + // "sorts":["products.count desc 0"], + // "limit":"500", + // "query_timezone":"America/Los_Angeles" + // } + // ``` + // + // When using the Ruby SDK this would be passed as a Ruby hash like: + // ``` + // { + // :model=>"thelook", + // :view=>"inventory_items", + // :fields=> + // ["category.name", + // "inventory_items.days_in_inventory_tier", + // "products.count"], + // :filters=>{:"category.name"=>"socks"}, + // :sorts=>["products.count desc 0"], + // :limit=>"500", + // :query_timezone=>"America/Los_Angeles", + // } + // ``` + // + // This will return the result of running the query in the format specified by the 'result_format' parameter. + // + // Supported formats: + // + // | result_format | Description + // | :-----------: | :--- | + // | json | Plain json + // | json_detail | Row data plus metadata describing the fields, pivots, table calcs, and other aspects of the query + // | csv | Comma separated values with a header + // | txt | Tab separated values with a header + // | html | Simple html + // | md | Simple markdown + // | xlsx | MS Excel spreadsheet + // | sql | Returns the generated SQL rather than running the query + // | png | A PNG image of the visualization of the query + // | jpg | A JPG image of the visualization of the query + // + // + // + rpc RunInlineQuery(RunInlineQueryReq) returns (stream RunInlineQueryResponse); + + // ### Run an URL encoded query. + // + // This requires the caller to encode the specifiers for the query into the URL query part using + // Looker-specific syntax as explained below. + // + // Generally, you would want to use one of the methods that takes the parameters as json in the POST body + // for creating and/or running queries. This method exists for cases where one really needs to encode the + // parameters into the URL of a single 'GET' request. This matches the way that the Looker UI formats + // 'explore' URLs etc. + // + // The parameters here are very similar to the json body formatting except that the filter syntax is + // tricky. Unfortunately, this format makes this method not currently callable via the 'Try it out!' button + // in this documentation page. But, this is callable when creating URLs manually or when using the Looker SDK. + // + // Here is an example inline query URL: + // + // ``` + // https://looker.mycompany.com:19999/api/3.0/queries/models/thelook/views/inventory_items/run/json?fields=category.name,inventory_items.days_in_inventory_tier,products.count&f[category.name]=socks&sorts=products.count+desc+0&limit=500&query_timezone=America/Los_Angeles + // ``` + // + // When invoking this endpoint with the Ruby SDK, pass the query parameter parts as a hash. The hash to match the above would look like: + // + // ```ruby + // query_params = + // { + // fields: "category.name,inventory_items.days_in_inventory_tier,products.count", + // :"f[category.name]" => "socks", + // sorts: "products.count desc 0", + // limit: "500", + // query_timezone: "America/Los_Angeles" + // } + // response = ruby_sdk.run_url_encoded_query('thelook','inventory_items','json', query_params) + // + // ``` + // + // Again, it is generally easier to use the variant of this method that passes the full query in the POST body. + // This method is available for cases where other alternatives won't fit the need. + // + // Supported formats: + // + // | result_format | Description + // | :-----------: | :--- | + // | json | Plain json + // | json_detail | Row data plus metadata describing the fields, pivots, table calcs, and other aspects of the query + // | csv | Comma separated values with a header + // | txt | Tab separated values with a header + // | html | Simple html + // | md | Simple markdown + // | xlsx | MS Excel spreadsheet + // | sql | Returns the generated SQL rather than running the query + // | png | A PNG image of the visualization of the query + // | jpg | A JPG image of the visualization of the query + // + // + // + rpc RunUrlEncodedQuery(RunUrlEncodedQueryReq) returns (stream RunUrlEncodedQueryResponse); + + // ### Get Merge Query + // + // Returns a merge query object given its id. + // + rpc MergeQuery(MergeQueryReq) returns (stream MergeQueryResponse); + + // ### Create Merge Query + // + // Creates a new merge query object. + // + // A merge query takes the results of one or more queries and combines (merges) the results + // according to field mapping definitions. The result is similar to a SQL left outer join. + // + // A merge query can merge results of queries from different SQL databases. + // + // The order that queries are defined in the source_queries array property is significant. The + // first query in the array defines the primary key into which the results of subsequent + // queries will be merged. + // + // Like model/view query objects, merge queries are immutable and have structural identity - if + // you make a request to create a new merge query that is identical to an existing merge query, + // the existing merge query will be returned instead of creating a duplicate. Conversely, any + // change to the contents of a merge query will produce a new object with a new id. + // + rpc CreateMergeQuery(CreateMergeQueryReq) returns (stream CreateMergeQueryResponse); + + // Get information about all running queries. + // + rpc AllRunningQueries(AllRunningQueriesReq) returns (stream AllRunningQueriesStreamResponse); + + // Kill a query with a specific query_task_id. + // + rpc KillQuery(KillQueryReq) returns (stream KillQueryResponse); + + // Get a SQL Runner query. + rpc SqlQuery(SqlQueryReq) returns (stream SqlQueryResponse); + + // ### Create a SQL Runner Query + // + // Either the `connection_name` or `model_name` parameter MUST be provided. + // + rpc CreateSqlQuery(CreateSqlQueryReq) returns (stream CreateSqlQueryResponse); + + // Execute a SQL Runner query in a given result_format. + rpc RunSqlQuery(RunSqlQueryReq) returns (stream RunSqlQueryResponse); + + + + // RenderTask: Manage Render Tasks + + // ### Create a new task to render a look to an image. + // + // Returns a render task object. + // To check the status of a render task, pass the render_task.id to [Get Render Task](#!/RenderTask/get_render_task). + // Once the render task is complete, you can download the resulting document or image using [Get Render Task Results](#!/RenderTask/get_render_task_results). + // + // + rpc CreateLookRenderTask(CreateLookRenderTaskReq) returns (stream CreateLookRenderTaskResponse); + + // ### Create a new task to render an existing query to an image. + // + // Returns a render task object. + // To check the status of a render task, pass the render_task.id to [Get Render Task](#!/RenderTask/get_render_task). + // Once the render task is complete, you can download the resulting document or image using [Get Render Task Results](#!/RenderTask/get_render_task_results). + // + // + rpc CreateQueryRenderTask(CreateQueryRenderTaskReq) returns (stream CreateQueryRenderTaskResponse); + + // ### Create a new task to render a dashboard to a document or image. + // + // Returns a render task object. + // To check the status of a render task, pass the render_task.id to [Get Render Task](#!/RenderTask/get_render_task). + // Once the render task is complete, you can download the resulting document or image using [Get Render Task Results](#!/RenderTask/get_render_task_results). + // + // + rpc CreateDashboardRenderTask(CreateDashboardRenderTaskReq) returns (stream CreateDashboardRenderTaskResponse); + + // ### Get information about a render task. + // + // Returns a render task object. + // To check the status of a render task, pass the render_task.id to [Get Render Task](#!/RenderTask/get_render_task). + // Once the render task is complete, you can download the resulting document or image using [Get Render Task Results](#!/RenderTask/get_render_task_results). + // + // + rpc RenderTask(RenderTaskReq) returns (stream RenderTaskResponse); + + // ### Get the document or image produced by a completed render task. + // + // Note that the PDF or image result will be a binary blob in the HTTP response, as indicated by the + // Content-Type in the response headers. This may require specialized (or at least different) handling than text + // responses such as JSON. You may need to tell your HTTP client that the response is binary so that it does not + // attempt to parse the binary data as text. + // + // If the render task exists but has not finished rendering the results, the response HTTP status will be + // **202 Accepted**, the response body will be empty, and the response will have a Retry-After header indicating + // that the caller should repeat the request at a later time. + // + // Returns 404 if the render task cannot be found, if the cached result has expired, or if the caller + // does not have permission to view the results. + // + // For detailed information about the status of the render task, use [Render Task](#!/RenderTask/render_task). + // Polling loops waiting for completion of a render task would be better served by polling **render_task(id)** until + // the task status reaches completion (or error) instead of polling **render_task_results(id)** alone. + // + rpc RenderTaskResults(RenderTaskResultsReq) returns (stream RenderTaskResultsResponse); + + // ### Create a new task to render a dashboard element to an image. + // + // Returns a render task object. + // To check the status of a render task, pass the render_task.id to [Get Render Task](#!/RenderTask/get_render_task). + // Once the render task is complete, you can download the resulting document or image using [Get Render Task Results](#!/RenderTask/get_render_task_results). + // + // + rpc CreateDashboardElementRenderTask(CreateDashboardElementRenderTaskReq) returns (stream CreateDashboardElementRenderTaskResponse); + + + + // Role: Manage Roles + + // ### Search model sets + // Returns all model set records that match the given search criteria. + // If multiple search params are given and `filter_or` is FALSE or not specified, + // search params are combined in a logical AND operation. + // Only rows that match *all* search param criteria will be returned. + // + // If `filter_or` is TRUE, multiple search params are combined in a logical OR operation. + // Results will include rows that match **any** of the search criteria. + // + // String search params use case-insensitive matching. + // String search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions. + // example="dan%" will match "danger" and "Danzig" but not "David" + // example="D_m%" will match "Damage" and "dump" + // + // Integer search params can accept a single value or a comma separated list of values. The multiple + // values will be combined under a logical OR operation - results will match at least one of + // the given values. + // + // Most search params can accept "IS NULL" and "NOT NULL" as special expressions to match + // or exclude (respectively) rows where the column is null. + // + // Boolean search params accept only "true" and "false" as values. + // + // + rpc SearchModelSets(SearchModelSetsReq) returns (stream SearchModelSetsStreamResponse); + + // ### Get information about the model set with a specific id. + // + rpc ModelSet(ModelSetReq) returns (stream ModelSetResponse); + + // ### Update information about the model set with a specific id. + // + rpc UpdateModelSet(UpdateModelSetReq) returns (stream UpdateModelSetResponse); + + // ### Delete the model set with a specific id. + // + rpc DeleteModelSet(DeleteModelSetReq) returns (stream DeleteModelSetResponse); + + // ### Get information about all model sets. + // + rpc AllModelSets(AllModelSetsReq) returns (stream AllModelSetsStreamResponse); + + // ### Create a model set with the specified information. Model sets are used by Roles. + // + rpc CreateModelSet(CreateModelSetReq) returns (stream CreateModelSetResponse); + + // ### Get all supported permissions. + // + rpc AllPermissions(AllPermissionsReq) returns (stream AllPermissionsStreamResponse); + + // ### Search permission sets + // Returns all permission set records that match the given search criteria. + // If multiple search params are given and `filter_or` is FALSE or not specified, + // search params are combined in a logical AND operation. + // Only rows that match *all* search param criteria will be returned. + // + // If `filter_or` is TRUE, multiple search params are combined in a logical OR operation. + // Results will include rows that match **any** of the search criteria. + // + // String search params use case-insensitive matching. + // String search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions. + // example="dan%" will match "danger" and "Danzig" but not "David" + // example="D_m%" will match "Damage" and "dump" + // + // Integer search params can accept a single value or a comma separated list of values. The multiple + // values will be combined under a logical OR operation - results will match at least one of + // the given values. + // + // Most search params can accept "IS NULL" and "NOT NULL" as special expressions to match + // or exclude (respectively) rows where the column is null. + // + // Boolean search params accept only "true" and "false" as values. + // + // + rpc SearchPermissionSets(SearchPermissionSetsReq) returns (stream SearchPermissionSetsStreamResponse); + + // ### Get information about the permission set with a specific id. + // + rpc PermissionSet(PermissionSetReq) returns (stream PermissionSetResponse); + + // ### Update information about the permission set with a specific id. + // + rpc UpdatePermissionSet(UpdatePermissionSetReq) returns (stream UpdatePermissionSetResponse); + + // ### Delete the permission set with a specific id. + // + rpc DeletePermissionSet(DeletePermissionSetReq) returns (stream DeletePermissionSetResponse); + + // ### Get information about all permission sets. + // + rpc AllPermissionSets(AllPermissionSetsReq) returns (stream AllPermissionSetsStreamResponse); + + // ### Create a permission set with the specified information. Permission sets are used by Roles. + // + rpc CreatePermissionSet(CreatePermissionSetReq) returns (stream CreatePermissionSetResponse); + + // ### Get information about all roles. + // + rpc AllRoles(AllRolesReq) returns (stream AllRolesStreamResponse); + + // ### Create a role with the specified information. + // + rpc CreateRole(CreateRoleReq) returns (stream CreateRoleResponse); + + // ### Search roles + // + // Returns all role records that match the given search criteria. + // + // If multiple search params are given and `filter_or` is FALSE or not specified, + // search params are combined in a logical AND operation. + // Only rows that match *all* search param criteria will be returned. + // + // If `filter_or` is TRUE, multiple search params are combined in a logical OR operation. + // Results will include rows that match **any** of the search criteria. + // + // String search params use case-insensitive matching. + // String search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions. + // example="dan%" will match "danger" and "Danzig" but not "David" + // example="D_m%" will match "Damage" and "dump" + // + // Integer search params can accept a single value or a comma separated list of values. The multiple + // values will be combined under a logical OR operation - results will match at least one of + // the given values. + // + // Most search params can accept "IS NULL" and "NOT NULL" as special expressions to match + // or exclude (respectively) rows where the column is null. + // + // Boolean search params accept only "true" and "false" as values. + // + // + rpc SearchRoles(SearchRolesReq) returns (stream SearchRolesStreamResponse); + + // ### Search roles include user count + // + // Returns all role records that match the given search criteria, and attaches + // associated user counts. + // + // If multiple search params are given and `filter_or` is FALSE or not specified, + // search params are combined in a logical AND operation. + // Only rows that match *all* search param criteria will be returned. + // + // If `filter_or` is TRUE, multiple search params are combined in a logical OR operation. + // Results will include rows that match **any** of the search criteria. + // + // String search params use case-insensitive matching. + // String search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions. + // example="dan%" will match "danger" and "Danzig" but not "David" + // example="D_m%" will match "Damage" and "dump" + // + // Integer search params can accept a single value or a comma separated list of values. The multiple + // values will be combined under a logical OR operation - results will match at least one of + // the given values. + // + // Most search params can accept "IS NULL" and "NOT NULL" as special expressions to match + // or exclude (respectively) rows where the column is null. + // + // Boolean search params accept only "true" and "false" as values. + // + // + rpc SearchRolesWithUserCount(SearchRolesWithUserCountReq) returns (stream SearchRolesWithUserCountStreamResponse); + + // ### Get information about the role with a specific id. + // + rpc Role(RoleReq) returns (stream RoleResponse); + + // ### Update information about the role with a specific id. + // + rpc UpdateRole(UpdateRoleReq) returns (stream UpdateRoleResponse); + + // ### Delete the role with a specific id. + // + rpc DeleteRole(DeleteRoleReq) returns (stream DeleteRoleResponse); + + // ### Get information about all the groups with the role that has a specific id. + // + rpc RoleGroups(RoleGroupsReq) returns (stream RoleGroupsStreamResponse); + + // ### Set all groups for a role, removing all existing group associations from that role. + // + rpc SetRoleGroups(SetRoleGroupsReq) returns (stream SetRoleGroupsStreamResponse); + + // ### Get information about all the users with the role that has a specific id. + // + rpc RoleUsers(RoleUsersReq) returns (stream RoleUsersStreamResponse); + + // ### Set all the users of the role with a specific id. + // + rpc SetRoleUsers(SetRoleUsersReq) returns (stream SetRoleUsersStreamResponse); + + + + // ScheduledPlan: Manage Scheduled Plans + + // ### Get Scheduled Plans for a Space + // + // Returns scheduled plans owned by the caller for a given space id. + // + rpc ScheduledPlansForSpace(ScheduledPlansForSpaceReq) returns (stream ScheduledPlansForSpaceStreamResponse); + + // ### Get Information About a Scheduled Plan + // + // Admins can fetch information about other users' Scheduled Plans. + // + rpc ScheduledPlan(ScheduledPlanReq) returns (stream ScheduledPlanResponse); + + // ### Update a Scheduled Plan + // + // Admins can update other users' Scheduled Plans. + // + // Note: Any scheduled plan destinations specified in an update will **replace** all scheduled plan destinations + // currently defined for the scheduled plan. + // + // For Example: If a scheduled plan has destinations A, B, and C, and you call update on this scheduled plan + // specifying only B in the destinations, then destinations A and C will be deleted by the update. + // + // Updating a scheduled plan to assign null or an empty array to the scheduled_plan_destinations property is an error, as a scheduled plan must always have at least one destination. + // + // If you omit the scheduled_plan_destinations property from the object passed to update, then the destinations + // defined on the original scheduled plan will remain unchanged. + // + // #### Email Permissions: + // + // For details about permissions required to schedule delivery to email and the safeguards + // Looker offers to protect against sending to unauthorized email destinations, see [Email Domain Whitelist for Scheduled Looks](https://docs.looker.com/r/api/embed-permissions). + // + // + // #### Scheduled Plan Destination Formats + // + // Scheduled plan destinations must specify the data format to produce and send to the destination. + // + // Formats: + // + // | format | Description + // | :-----------: | :--- | + // | json | A JSON object containing a `data` property which contains an array of JSON objects, one per row. No metadata. + // | json_detail | Row data plus metadata describing the fields, pivots, table calcs, and other aspects of the query + // | inline_json | Same as the JSON format, except that the `data` property is a string containing JSON-escaped row data. Additional properties describe the data operation. This format is primarily used to send data to web hooks so that the web hook doesn't have to re-encode the JSON row data in order to pass it on to its ultimate destination. + // | csv | Comma separated values with a header + // | txt | Tab separated values with a header + // | html | Simple html + // | xlsx | MS Excel spreadsheet + // | wysiwyg_pdf | Dashboard rendered in a tiled layout to produce a PDF document + // | assembled_pdf | Dashboard rendered in a single column layout to produce a PDF document + // | wysiwyg_png | Dashboard rendered in a tiled layout to produce a PNG image + // || + // + // Valid formats vary by destination type and source object. `wysiwyg_pdf` is only valid for dashboards, for example. + // + // + // + rpc UpdateScheduledPlan(UpdateScheduledPlanReq) returns (stream UpdateScheduledPlanResponse); + + // ### Delete a Scheduled Plan + // + // Normal users can only delete their own scheduled plans. + // Admins can delete other users' scheduled plans. + // This delete cannot be undone. + // + rpc DeleteScheduledPlan(DeleteScheduledPlanReq) returns (stream DeleteScheduledPlanResponse); + + // ### List All Scheduled Plans + // + // Returns all scheduled plans which belong to the caller or given user. + // + // If no user_id is provided, this function returns the scheduled plans owned by the caller. + // + // + // To list all schedules for all users, pass `all_users=true`. + // + // + // The caller must have `see_schedules` permission to see other users' scheduled plans. + // + // + // + rpc AllScheduledPlans(AllScheduledPlansReq) returns (stream AllScheduledPlansStreamResponse); + + // ### Create a Scheduled Plan + // + // Create a scheduled plan to render a Look or Dashboard on a recurring schedule. + // + // To create a scheduled plan, you MUST provide values for the following fields: + // `name` + // and + // `look_id`, `dashboard_id`, `lookml_dashboard_id`, or `query_id` + // and + // `cron_tab` or `datagroup` + // and + // at least one scheduled_plan_destination + // + // A scheduled plan MUST have at least one scheduled_plan_destination defined. + // + // When `look_id` is set, `require_no_results`, `require_results`, and `require_change` are all required. + // + // If `create_scheduled_plan` fails with a 422 error, be sure to look at the error messages in the response which will explain exactly what fields are missing or values that are incompatible. + // + // The queries that provide the data for the look or dashboard are run in the context of user account that owns the scheduled plan. + // + // When `run_as_recipient` is `false` or not specified, the queries that provide the data for the + // look or dashboard are run in the context of user account that owns the scheduled plan. + // + // When `run_as_recipient` is `true` and all the email recipients are Looker user accounts, the + // queries are run in the context of each recipient, so different recipients may see different + // data from the same scheduled render of a look or dashboard. For more details, see [Run As Recipient](https://looker.com/docs/r/admin/run-as-recipient). + // + // Admins can create and modify scheduled plans on behalf of other users by specifying a user id. + // Non-admin users may not create or modify scheduled plans by or for other users. + // + // #### Email Permissions: + // + // For details about permissions required to schedule delivery to email and the safeguards + // Looker offers to protect against sending to unauthorized email destinations, see [Email Domain Whitelist for Scheduled Looks](https://docs.looker.com/r/api/embed-permissions). + // + // + // #### Scheduled Plan Destination Formats + // + // Scheduled plan destinations must specify the data format to produce and send to the destination. + // + // Formats: + // + // | format | Description + // | :-----------: | :--- | + // | json | A JSON object containing a `data` property which contains an array of JSON objects, one per row. No metadata. + // | json_detail | Row data plus metadata describing the fields, pivots, table calcs, and other aspects of the query + // | inline_json | Same as the JSON format, except that the `data` property is a string containing JSON-escaped row data. Additional properties describe the data operation. This format is primarily used to send data to web hooks so that the web hook doesn't have to re-encode the JSON row data in order to pass it on to its ultimate destination. + // | csv | Comma separated values with a header + // | txt | Tab separated values with a header + // | html | Simple html + // | xlsx | MS Excel spreadsheet + // | wysiwyg_pdf | Dashboard rendered in a tiled layout to produce a PDF document + // | assembled_pdf | Dashboard rendered in a single column layout to produce a PDF document + // | wysiwyg_png | Dashboard rendered in a tiled layout to produce a PNG image + // || + // + // Valid formats vary by destination type and source object. `wysiwyg_pdf` is only valid for dashboards, for example. + // + // + // + rpc CreateScheduledPlan(CreateScheduledPlanReq) returns (stream CreateScheduledPlanResponse); + + // ### Run a Scheduled Plan Immediately + // + // Create a scheduled plan that runs only once, and immediately. + // + // This can be useful for testing a Scheduled Plan before committing to a production schedule. + // + // Admins can create scheduled plans on behalf of other users by specifying a user id. + // + // This API is rate limited to prevent it from being used for relay spam or DoS attacks + // + // #### Email Permissions: + // + // For details about permissions required to schedule delivery to email and the safeguards + // Looker offers to protect against sending to unauthorized email destinations, see [Email Domain Whitelist for Scheduled Looks](https://docs.looker.com/r/api/embed-permissions). + // + // + // #### Scheduled Plan Destination Formats + // + // Scheduled plan destinations must specify the data format to produce and send to the destination. + // + // Formats: + // + // | format | Description + // | :-----------: | :--- | + // | json | A JSON object containing a `data` property which contains an array of JSON objects, one per row. No metadata. + // | json_detail | Row data plus metadata describing the fields, pivots, table calcs, and other aspects of the query + // | inline_json | Same as the JSON format, except that the `data` property is a string containing JSON-escaped row data. Additional properties describe the data operation. This format is primarily used to send data to web hooks so that the web hook doesn't have to re-encode the JSON row data in order to pass it on to its ultimate destination. + // | csv | Comma separated values with a header + // | txt | Tab separated values with a header + // | html | Simple html + // | xlsx | MS Excel spreadsheet + // | wysiwyg_pdf | Dashboard rendered in a tiled layout to produce a PDF document + // | assembled_pdf | Dashboard rendered in a single column layout to produce a PDF document + // | wysiwyg_png | Dashboard rendered in a tiled layout to produce a PNG image + // || + // + // Valid formats vary by destination type and source object. `wysiwyg_pdf` is only valid for dashboards, for example. + // + // + // + rpc ScheduledPlanRunOnce(ScheduledPlanRunOnceReq) returns (stream ScheduledPlanRunOnceResponse); + + // ### Get Scheduled Plans for a Look + // + // Returns all scheduled plans for a look which belong to the caller or given user. + // + // If no user_id is provided, this function returns the scheduled plans owned by the caller. + // + // + // To list all schedules for all users, pass `all_users=true`. + // + // + // The caller must have `see_schedules` permission to see other users' scheduled plans. + // + // + // + rpc ScheduledPlansForLook(ScheduledPlansForLookReq) returns (stream ScheduledPlansForLookStreamResponse); + + // ### Get Scheduled Plans for a Dashboard + // + // Returns all scheduled plans for a dashboard which belong to the caller or given user. + // + // If no user_id is provided, this function returns the scheduled plans owned by the caller. + // + // + // To list all schedules for all users, pass `all_users=true`. + // + // + // The caller must have `see_schedules` permission to see other users' scheduled plans. + // + // + // + rpc ScheduledPlansForDashboard(ScheduledPlansForDashboardReq) returns (stream ScheduledPlansForDashboardStreamResponse); + + // ### Get Scheduled Plans for a LookML Dashboard + // + // Returns all scheduled plans for a LookML Dashboard which belong to the caller or given user. + // + // If no user_id is provided, this function returns the scheduled plans owned by the caller. + // + // + // To list all schedules for all users, pass `all_users=true`. + // + // + // The caller must have `see_schedules` permission to see other users' scheduled plans. + // + // + // + rpc ScheduledPlansForLookmlDashboard(ScheduledPlansForLookmlDashboardReq) returns (stream ScheduledPlansForLookmlDashboardStreamResponse); + + // ### Run a Scheduled Plan By Id Immediately + // This function creates a run-once schedule plan based on an existing scheduled plan, + // applies modifications (if any) to the new scheduled plan, and runs the new schedule plan immediately. + // This can be useful for testing modifications to an existing scheduled plan before committing to a production schedule. + // + // This function internally performs the following operations: + // + // 1. Copies the properties of the existing scheduled plan into a new scheduled plan + // 2. Copies any properties passed in the JSON body of this request into the new scheduled plan (replacing the original values) + // 3. Creates the new scheduled plan + // 4. Runs the new scheduled plan + // + // The original scheduled plan is not modified by this operation. + // Admins can create, modify, and run scheduled plans on behalf of other users by specifying a user id. + // Non-admins can only create, modify, and run their own scheduled plans. + // + // #### Email Permissions: + // + // For details about permissions required to schedule delivery to email and the safeguards + // Looker offers to protect against sending to unauthorized email destinations, see [Email Domain Whitelist for Scheduled Looks](https://docs.looker.com/r/api/embed-permissions). + // + // + // #### Scheduled Plan Destination Formats + // + // Scheduled plan destinations must specify the data format to produce and send to the destination. + // + // Formats: + // + // | format | Description + // | :-----------: | :--- | + // | json | A JSON object containing a `data` property which contains an array of JSON objects, one per row. No metadata. + // | json_detail | Row data plus metadata describing the fields, pivots, table calcs, and other aspects of the query + // | inline_json | Same as the JSON format, except that the `data` property is a string containing JSON-escaped row data. Additional properties describe the data operation. This format is primarily used to send data to web hooks so that the web hook doesn't have to re-encode the JSON row data in order to pass it on to its ultimate destination. + // | csv | Comma separated values with a header + // | txt | Tab separated values with a header + // | html | Simple html + // | xlsx | MS Excel spreadsheet + // | wysiwyg_pdf | Dashboard rendered in a tiled layout to produce a PDF document + // | assembled_pdf | Dashboard rendered in a single column layout to produce a PDF document + // | wysiwyg_png | Dashboard rendered in a tiled layout to produce a PNG image + // || + // + // Valid formats vary by destination type and source object. `wysiwyg_pdf` is only valid for dashboards, for example. + // + // + // + // This API is rate limited to prevent it from being used for relay spam or DoS attacks + // + // + rpc ScheduledPlanRunOnceById(ScheduledPlanRunOnceByIdReq) returns (stream ScheduledPlanRunOnceByIdResponse); + + + + // Session: Session Information + + // ### Get API Session + // + // Returns information about the current API session, such as which workspace is selected for the session. + // + rpc Session(SessionReq) returns (stream SessionResponse); + + // ### Update API Session + // + // #### API Session Workspace + // + // You can use this endpoint to change the active workspace for the current API session. + // + // Only one workspace can be active in a session. The active workspace can be changed + // any number of times in a session. + // + // The default workspace for API sessions is the "production" workspace. + // + // All Looker APIs that use projects or lookml models (such as running queries) will + // use the version of project and model files defined by this workspace for the lifetime of the + // current API session or until the session workspace is changed again. + // + // An API session has the same lifetime as the access_token used to authenticate API requests. Each successful + // API login generates a new access_token and a new API session. + // + // If your Looker API client application needs to work in a dev workspace across multiple + // API sessions, be sure to select the dev workspace after each login. + // + rpc UpdateSession(UpdateSessionReq) returns (stream UpdateSessionResponse); + + + + // Theme: Manage Themes + + // ### Get an array of all existing themes + // + // Get a **single theme** by id with [Theme](#!/Theme/theme) + // + // This method returns an array of all existing themes. The active time for the theme is not considered. + // + // **Note**: Custom themes needs to be enabled by Looker. Unless custom themes are enabled, only the automatically generated default theme can be used. Please contact your Account Manager or help.looker.com to update your license for this feature. + // + // + rpc AllThemes(AllThemesReq) returns (stream AllThemesStreamResponse); + + // ### Create a theme + // + // Creates a new theme object, returning the theme details, including the created id. + // + // If `settings` are not specified, the default theme settings will be copied into the new theme. + // + // The theme `name` can only contain alphanumeric characters or underscores. Theme names should not contain any confidential information, such as customer names. + // + // **Update** an existing theme with [Update Theme](#!/Theme/update_theme) + // + // **Permanently delete** an existing theme with [Delete Theme](#!/Theme/delete_theme) + // + // For more information, see [Creating and Applying Themes](https://looker.com/docs/r/admin/themes). + // + // **Note**: Custom themes needs to be enabled by Looker. Unless custom themes are enabled, only the automatically generated default theme can be used. Please contact your Account Manager or help.looker.com to update your license for this feature. + // + // + rpc CreateTheme(CreateThemeReq) returns (stream CreateThemeResponse); + + // ### Search all themes for matching criteria. + // + // Returns an **array of theme objects** that match the specified search criteria. + // + // | Search Parameters | Description + // | :-------------------: | :------ | + // | `begin_at` only | Find themes active at or after `begin_at` + // | `end_at` only | Find themes active at or before `end_at` + // | both set | Find themes with an active inclusive period between `begin_at` and `end_at` + // + // Note: Range matching requires boolean AND logic. + // When using `begin_at` and `end_at` together, do not use `filter_or`=TRUE + // + // If multiple search params are given and `filter_or` is FALSE or not specified, + // search params are combined in a logical AND operation. + // Only rows that match *all* search param criteria will be returned. + // + // If `filter_or` is TRUE, multiple search params are combined in a logical OR operation. + // Results will include rows that match **any** of the search criteria. + // + // String search params use case-insensitive matching. + // String search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions. + // example="dan%" will match "danger" and "Danzig" but not "David" + // example="D_m%" will match "Damage" and "dump" + // + // Integer search params can accept a single value or a comma separated list of values. The multiple + // values will be combined under a logical OR operation - results will match at least one of + // the given values. + // + // Most search params can accept "IS NULL" and "NOT NULL" as special expressions to match + // or exclude (respectively) rows where the column is null. + // + // Boolean search params accept only "true" and "false" as values. + // + // + // Get a **single theme** by id with [Theme](#!/Theme/theme) + // + // **Note**: Custom themes needs to be enabled by Looker. Unless custom themes are enabled, only the automatically generated default theme can be used. Please contact your Account Manager or help.looker.com to update your license for this feature. + // + // + rpc SearchThemes(SearchThemesReq) returns (stream SearchThemesStreamResponse); + + // ### Get the default theme + // + // Returns the active theme object set as the default. + // + // The **default** theme name can be set in the UI on the Admin|Theme UI page + // + // The optional `ts` parameter can specify a different timestamp than "now." If specified, it returns the default theme at the time indicated. + // + rpc DefaultTheme(DefaultThemeReq) returns (stream DefaultThemeResponse); + + // ### Set the global default theme by theme name + // + // Only Admin users can call this function. + // + // Only an active theme with no expiration (`end_at` not set) can be assigned as the default theme. As long as a theme has an active record with no expiration, it can be set as the default. + // + // [Create Theme](#!/Theme/create) has detailed information on rules for default and active themes + // + // Returns the new specified default theme object. + // + // **Note**: Custom themes needs to be enabled by Looker. Unless custom themes are enabled, only the automatically generated default theme can be used. Please contact your Account Manager or help.looker.com to update your license for this feature. + // + // + rpc SetDefaultTheme(SetDefaultThemeReq) returns (stream SetDefaultThemeResponse); + + // ### Get active themes + // + // Returns an array of active themes. + // + // If the `name` parameter is specified, it will return an array with one theme if it's active and found. + // + // The optional `ts` parameter can specify a different timestamp than "now." + // + // **Note**: Custom themes needs to be enabled by Looker. Unless custom themes are enabled, only the automatically generated default theme can be used. Please contact your Account Manager or help.looker.com to update your license for this feature. + // + // + // + rpc ActiveThemes(ActiveThemesReq) returns (stream ActiveThemesStreamResponse); + + // ### Get the named theme if it's active. Otherwise, return the default theme + // + // The optional `ts` parameter can specify a different timestamp than "now." + // Note: API users with `show` ability can call this function + // + // **Note**: Custom themes needs to be enabled by Looker. Unless custom themes are enabled, only the automatically generated default theme can be used. Please contact your Account Manager or help.looker.com to update your license for this feature. + // + // + rpc ThemeOrDefault(ThemeOrDefaultReq) returns (stream ThemeOrDefaultResponse); + + // ### Validate a theme with the specified information + // + // Validates all values set for the theme, returning any errors encountered, or 200 OK if valid + // + // See [Create Theme](#!/Theme/create_theme) for constraints + // + // **Note**: Custom themes needs to be enabled by Looker. Unless custom themes are enabled, only the automatically generated default theme can be used. Please contact your Account Manager or help.looker.com to update your license for this feature. + // + // + rpc ValidateTheme(ValidateThemeReq) returns (stream ValidateThemeResponse); + + // ### Get a theme by ID + // + // Use this to retrieve a specific theme, whether or not it's currently active. + // + // **Note**: Custom themes needs to be enabled by Looker. Unless custom themes are enabled, only the automatically generated default theme can be used. Please contact your Account Manager or help.looker.com to update your license for this feature. + // + // + rpc Theme(ThemeReq) returns (stream ThemeResponse); + + // ### Update the theme by id. + // + // **Note**: Custom themes needs to be enabled by Looker. Unless custom themes are enabled, only the automatically generated default theme can be used. Please contact your Account Manager or help.looker.com to update your license for this feature. + // + // + rpc UpdateTheme(UpdateThemeReq) returns (stream UpdateThemeResponse); + + // ### Delete a specific theme by id + // + // This operation permanently deletes the identified theme from the database. + // + // Because multiple themes can have the same name (with different activation time spans) themes can only be deleted by ID. + // + // All IDs associated with a theme name can be retrieved by searching for the theme name with [Theme Search](#!/Theme/search). + // + // **Note**: Custom themes needs to be enabled by Looker. Unless custom themes are enabled, only the automatically generated default theme can be used. Please contact your Account Manager or help.looker.com to update your license for this feature. + // + // + rpc DeleteTheme(DeleteThemeReq) returns (stream DeleteThemeResponse); + + + + // User: Manage Users + + // ### Search email credentials + // + // Returns all credentials_email records that match the given search criteria. + // + // If multiple search params are given and `filter_or` is FALSE or not specified, + // search params are combined in a logical AND operation. + // Only rows that match *all* search param criteria will be returned. + // + // If `filter_or` is TRUE, multiple search params are combined in a logical OR operation. + // Results will include rows that match **any** of the search criteria. + // + // String search params use case-insensitive matching. + // String search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions. + // example="dan%" will match "danger" and "Danzig" but not "David" + // example="D_m%" will match "Damage" and "dump" + // + // Integer search params can accept a single value or a comma separated list of values. The multiple + // values will be combined under a logical OR operation - results will match at least one of + // the given values. + // + // Most search params can accept "IS NULL" and "NOT NULL" as special expressions to match + // or exclude (respectively) rows where the column is null. + // + // Boolean search params accept only "true" and "false" as values. + // + // + rpc SearchCredentialsEmail(SearchCredentialsEmailReq) returns (stream SearchCredentialsEmailStreamResponse); + + // ### Get information about the current user; i.e. the user account currently calling the API. + // + rpc Me(MeReq) returns (stream MeResponse); + + // ### Get information about all users. + // + rpc AllUsers(AllUsersReq) returns (stream AllUsersStreamResponse); + + // ### Create a user with the specified information. + // + rpc CreateUser(CreateUserReq) returns (stream CreateUserResponse); + + // ### Search users + // + // Returns all* user records that match the given search criteria. + // + // If multiple search params are given and `filter_or` is FALSE or not specified, + // search params are combined in a logical AND operation. + // Only rows that match *all* search param criteria will be returned. + // + // If `filter_or` is TRUE, multiple search params are combined in a logical OR operation. + // Results will include rows that match **any** of the search criteria. + // + // String search params use case-insensitive matching. + // String search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions. + // example="dan%" will match "danger" and "Danzig" but not "David" + // example="D_m%" will match "Damage" and "dump" + // + // Integer search params can accept a single value or a comma separated list of values. The multiple + // values will be combined under a logical OR operation - results will match at least one of + // the given values. + // + // Most search params can accept "IS NULL" and "NOT NULL" as special expressions to match + // or exclude (respectively) rows where the column is null. + // + // Boolean search params accept only "true" and "false" as values. + // + // + // (*) Results are always filtered to the level of information the caller is permitted to view. + // Looker admins can see all user details; normal users in an open system can see + // names of other users but no details; normal users in a closed system can only see + // names of other users who are members of the same group as the user. + // + // + rpc SearchUsers(SearchUsersReq) returns (stream SearchUsersStreamResponse); + + // ### Search for user accounts by name + // + // Returns all user accounts where `first_name` OR `last_name` OR `email` field values match a pattern. + // The pattern can contain `%` and `_` wildcards as in SQL LIKE expressions. + // + // Any additional search params will be combined into a logical AND expression. + // + rpc SearchUsersNames(SearchUsersNamesReq) returns (stream SearchUsersNamesStreamResponse); + + // ### Get information about the user with a specific id. + // + // If the caller is an admin or the caller is the user being specified, then full user information will + // be returned. Otherwise, a minimal 'public' variant of the user information will be returned. This contains + // The user name and avatar url, but no sensitive information. + // + rpc User(UserReq) returns (stream UserResponse); + + // ### Update information about the user with a specific id. + // + rpc UpdateUser(UpdateUserReq) returns (stream UpdateUserResponse); + + // ### Delete the user with a specific id. + // + // **DANGER** this will delete the user and all looks and other information owned by the user. + // + rpc DeleteUser(DeleteUserReq) returns (stream DeleteUserResponse); + + // ### Get information about the user with a credential of given type with specific id. + // + // This is used to do things like find users by their embed external_user_id. Or, find the user with + // a given api3 client_id, etc. The 'credential_type' matches the 'type' name of the various credential + // types. It must be one of the values listed in the table below. The 'credential_id' is your unique Id + // for the user and is specific to each type of credential. + // + // An example using the Ruby sdk might look like: + // + // `sdk.user_for_credential('embed', 'customer-4959425')` + // + // This table shows the supported 'Credential Type' strings. The right column is for reference; it shows + // which field in the given credential type is actually searched when finding a user with the supplied + // 'credential_id'. + // + // | Credential Types | Id Field Matched | + // | ---------------- | ---------------- | + // | email | email | + // | google | google_user_id | + // | saml | saml_user_id | + // | oidc | oidc_user_id | + // | ldap | ldap_id | + // | api | token | + // | api3 | client_id | + // | embed | external_user_id | + // | looker_openid | email | + // + // **NOTE**: The 'api' credential type was only used with the legacy Looker query API and is no longer supported. The credential type for API you are currently looking at is 'api3'. + // + // + rpc UserForCredential(UserForCredentialReq) returns (stream UserForCredentialResponse); + + // ### Email/password login information for the specified user. + rpc UserCredentialsEmail(UserCredentialsEmailReq) returns (stream UserCredentialsEmailResponse); + + // ### Email/password login information for the specified user. + rpc CreateUserCredentialsEmail(CreateUserCredentialsEmailReq) returns (stream CreateUserCredentialsEmailResponse); + + // ### Email/password login information for the specified user. + rpc UpdateUserCredentialsEmail(UpdateUserCredentialsEmailReq) returns (stream UpdateUserCredentialsEmailResponse); + + // ### Email/password login information for the specified user. + rpc DeleteUserCredentialsEmail(DeleteUserCredentialsEmailReq) returns (stream DeleteUserCredentialsEmailResponse); + + // ### Two-factor login information for the specified user. + rpc UserCredentialsTotp(UserCredentialsTotpReq) returns (stream UserCredentialsTotpResponse); + + // ### Two-factor login information for the specified user. + rpc CreateUserCredentialsTotp(CreateUserCredentialsTotpReq) returns (stream CreateUserCredentialsTotpResponse); + + // ### Two-factor login information for the specified user. + rpc DeleteUserCredentialsTotp(DeleteUserCredentialsTotpReq) returns (stream DeleteUserCredentialsTotpResponse); + + // ### LDAP login information for the specified user. + rpc UserCredentialsLdap(UserCredentialsLdapReq) returns (stream UserCredentialsLdapResponse); + + // ### LDAP login information for the specified user. + rpc DeleteUserCredentialsLdap(DeleteUserCredentialsLdapReq) returns (stream DeleteUserCredentialsLdapResponse); + + // ### Google authentication login information for the specified user. + rpc UserCredentialsGoogle(UserCredentialsGoogleReq) returns (stream UserCredentialsGoogleResponse); + + // ### Google authentication login information for the specified user. + rpc DeleteUserCredentialsGoogle(DeleteUserCredentialsGoogleReq) returns (stream DeleteUserCredentialsGoogleResponse); + + // ### Saml authentication login information for the specified user. + rpc UserCredentialsSaml(UserCredentialsSamlReq) returns (stream UserCredentialsSamlResponse); + + // ### Saml authentication login information for the specified user. + rpc DeleteUserCredentialsSaml(DeleteUserCredentialsSamlReq) returns (stream DeleteUserCredentialsSamlResponse); + + // ### OpenID Connect (OIDC) authentication login information for the specified user. + rpc UserCredentialsOidc(UserCredentialsOidcReq) returns (stream UserCredentialsOidcResponse); + + // ### OpenID Connect (OIDC) authentication login information for the specified user. + rpc DeleteUserCredentialsOidc(DeleteUserCredentialsOidcReq) returns (stream DeleteUserCredentialsOidcResponse); + + // ### API 3 login information for the specified user. This is for the newer API keys that can be added for any user. + rpc UserCredentialsApi3(UserCredentialsApi3Req) returns (stream UserCredentialsApi3Response); + + // ### API 3 login information for the specified user. This is for the newer API keys that can be added for any user. + rpc DeleteUserCredentialsApi3(DeleteUserCredentialsApi3Req) returns (stream DeleteUserCredentialsApi3Response); + + // ### API 3 login information for the specified user. This is for the newer API keys that can be added for any user. + rpc AllUserCredentialsApi3s(AllUserCredentialsApi3sReq) returns (stream AllUserCredentialsApi3sStreamResponse); + + // ### API 3 login information for the specified user. This is for the newer API keys that can be added for any user. + rpc CreateUserCredentialsApi3(CreateUserCredentialsApi3Req) returns (stream CreateUserCredentialsApi3Response); + + // ### Embed login information for the specified user. + rpc UserCredentialsEmbed(UserCredentialsEmbedReq) returns (stream UserCredentialsEmbedResponse); + + // ### Embed login information for the specified user. + rpc DeleteUserCredentialsEmbed(DeleteUserCredentialsEmbedReq) returns (stream DeleteUserCredentialsEmbedResponse); + + // ### Embed login information for the specified user. + rpc AllUserCredentialsEmbeds(AllUserCredentialsEmbedsReq) returns (stream AllUserCredentialsEmbedsStreamResponse); + + // ### Looker Openid login information for the specified user. Used by Looker Analysts. + rpc UserCredentialsLookerOpenid(UserCredentialsLookerOpenidReq) returns (stream UserCredentialsLookerOpenidResponse); + + // ### Looker Openid login information for the specified user. Used by Looker Analysts. + rpc DeleteUserCredentialsLookerOpenid(DeleteUserCredentialsLookerOpenidReq) returns (stream DeleteUserCredentialsLookerOpenidResponse); + + // ### Web login session for the specified user. + rpc UserSession(UserSessionReq) returns (stream UserSessionResponse); + + // ### Web login session for the specified user. + rpc DeleteUserSession(DeleteUserSessionReq) returns (stream DeleteUserSessionResponse); + + // ### Web login session for the specified user. + rpc AllUserSessions(AllUserSessionsReq) returns (stream AllUserSessionsStreamResponse); + + // ### Create a password reset token. + // This will create a cryptographically secure random password reset token for the user. + // If the user already has a password reset token then this invalidates the old token and creates a new one. + // The token is expressed as the 'password_reset_url' of the user's email/password credential object. + // This takes an optional 'expires' param to indicate if the new token should be an expiring token. + // Tokens that expire are typically used for self-service password resets for existing users. + // Invitation emails for new users typically are not set to expire. + // The expire period is always 60 minutes when expires is enabled. + // This method can be called with an empty body. + // + rpc CreateUserCredentialsEmailPasswordReset(CreateUserCredentialsEmailPasswordResetReq) returns (stream CreateUserCredentialsEmailPasswordResetResponse); + + // ### Get information about roles of a given user + // + rpc UserRoles(UserRolesReq) returns (stream UserRolesStreamResponse); + + // ### Set roles of the user with a specific id. + // + rpc SetUserRoles(SetUserRolesReq) returns (stream SetUserRolesStreamResponse); + + // ### Get user attribute values for a given user. + // + // Returns the values of specified user attributes (or all user attributes) for a certain user. + // + // A value for each user attribute is searched for in the following locations, in this order: + // + // 1. in the user's account information + // 1. in groups that the user is a member of + // 1. the default value of the user attribute + // + // If more than one group has a value defined for a user attribute, the group with the lowest rank wins. + // + // The response will only include user attributes for which values were found. Use `include_unset=true` to include + // empty records for user attributes with no value. + // + // The value of all hidden user attributes will be blank. + // + rpc UserAttributeUserValues(UserAttributeUserValuesReq) returns (stream UserAttributeUserValuesStreamResponse); + + // ### Store a custom value for a user attribute in a user's account settings. + // + // Per-user user attribute values take precedence over group or default values. + // + rpc SetUserAttributeUserValue(SetUserAttributeUserValueReq) returns (stream SetUserAttributeUserValueResponse); + + // ### Delete a user attribute value from a user's account settings. + // + // After the user attribute value is deleted from the user's account settings, subsequent requests + // for the user attribute value for this user will draw from the user's groups or the default + // value of the user attribute. See [Get User Attribute Values](#!/User/user_attribute_user_values) for more + // information about how user attribute values are resolved. + // + rpc DeleteUserAttributeUserValue(DeleteUserAttributeUserValueReq) returns (stream DeleteUserAttributeUserValueResponse); + + // ### Send a password reset token. + // This will send a password reset email to the user. If a password reset token does not already exist + // for this user, it will create one and then send it. + // If the user has not yet set up their account, it will send a setup email to the user. + // The URL sent in the email is expressed as the 'password_reset_url' of the user's email/password credential object. + // Password reset URLs will expire in 60 minutes. + // This method can be called with an empty body. + // + rpc SendUserCredentialsEmailPasswordReset(SendUserCredentialsEmailPasswordResetReq) returns (stream SendUserCredentialsEmailPasswordResetResponse); + + // ### Change a disabled user's email addresses + // + // Allows the admin to change the email addresses for all the user's + // associated credentials. Will overwrite all associated email addresses with + // the value supplied in the 'email' body param. + // The user's 'is_disabled' status must be true. + // + rpc WipeoutUserEmails(WipeoutUserEmailsReq) returns (stream WipeoutUserEmailsResponse); + + // Create an embed user from an external user ID + // + rpc CreateEmbedUser(CreateEmbedUserReq) returns (stream CreateEmbedUserResponse); + + + + // UserAttribute: Manage User Attributes + + // ### Get information about all user attributes. + // + rpc AllUserAttributes(AllUserAttributesReq) returns (stream AllUserAttributesStreamResponse); + + // ### Create a new user attribute + // + // Permission information for a user attribute is conveyed through the `can` and `user_can_edit` fields. + // The `user_can_edit` field indicates whether an attribute is user-editable _anywhere_ in the application. + // The `can` field gives more granular access information, with the `set_value` child field indicating whether + // an attribute's value can be set by [Setting the User Attribute User Value](#!/User/set_user_attribute_user_value). + // + // Note: `name` and `label` fields must be unique across all user attributes in the Looker instance. + // Attempting to create a new user attribute with a name or label that duplicates an existing + // user attribute will fail with a 422 error. + // + rpc CreateUserAttribute(CreateUserAttributeReq) returns (stream CreateUserAttributeResponse); + + // ### Get information about a user attribute. + // + rpc UserAttribute(UserAttributeReq) returns (stream UserAttributeResponse); + + // ### Update a user attribute definition. + // + rpc UpdateUserAttribute(UpdateUserAttributeReq) returns (stream UpdateUserAttributeResponse); + + // ### Delete a user attribute (admin only). + // + rpc DeleteUserAttribute(DeleteUserAttributeReq) returns (stream DeleteUserAttributeResponse); + + // ### Returns all values of a user attribute defined by user groups, in precedence order. + // + // A user may be a member of multiple groups which define different values for a given user attribute. + // The order of group-values in the response determines precedence for selecting which group-value applies + // to a given user. For more information, see [Set User Attribute Group Values](#!/UserAttribute/set_user_attribute_group_values). + // + // Results will only include groups that the caller's user account has permission to see. + // + rpc AllUserAttributeGroupValues(AllUserAttributeGroupValuesReq) returns (stream AllUserAttributeGroupValuesStreamResponse); + + // ### Define values for a user attribute across a set of groups, in priority order. + // + // This function defines all values for a user attribute defined by user groups. This is a global setting, potentially affecting + // all users in the system. This function replaces any existing group value definitions for the indicated user attribute. + // + // The value of a user attribute for a given user is determined by searching the following locations, in this order: + // + // 1. the user's account settings + // 2. the groups that the user is a member of + // 3. the default value of the user attribute, if any + // + // The user may be a member of multiple groups which define different values for that user attribute. The order of items in the group_values parameter + // determines which group takes priority for that user. Lowest array index wins. + // + // An alternate method to indicate the selection precedence of group-values is to assign numbers to the 'rank' property of each + // group-value object in the array. Lowest 'rank' value wins. If you use this technique, you must assign a + // rank value to every group-value object in the array. + // + // To set a user attribute value for a single user, see [Set User Attribute User Value](#!/User/set_user_attribute_user_value). + // To set a user attribute value for all members of a group, see [Set User Attribute Group Value](#!/Group/update_user_attribute_group_value). + // + rpc SetUserAttributeGroupValues(SetUserAttributeGroupValuesReq) returns (stream SetUserAttributeGroupValuesStreamResponse); + + + + // Workspace: Manage Workspaces + + // ### Get All Workspaces + // + // Returns all workspaces available to the calling user. + // + rpc AllWorkspaces(AllWorkspacesReq) returns (stream AllWorkspacesStreamResponse); + + // ### Get A Workspace + // + // Returns information about a workspace such as the git status and selected branches + // of all projects available to the caller's user account. + // + // A workspace defines which versions of project files will be used to evaluate expressions + // and operations that use model definitions - operations such as running queries or rendering dashboards. + // Each project has its own git repository, and each project in a workspace may be configured to reference + // particular branch or revision within their respective repositories. + // + // There are two predefined workspaces available: "production" and "dev". + // + // The production workspace is shared across all Looker users. Models in the production workspace are read-only. + // Changing files in production is accomplished by modifying files in a git branch and using Pull Requests + // to merge the changes from the dev branch into the production branch, and then telling + // Looker to sync with production. + // + // The dev workspace is local to each Looker user. Changes made to project/model files in the dev workspace only affect + // that user, and only when the dev workspace is selected as the active workspace for the API session. + // (See set_session_workspace()). + // + // The dev workspace is NOT unique to an API session. Two applications accessing the Looker API using + // the same user account will see the same files in the dev workspace. To avoid collisions between + // API clients it's best to have each client login with API3 credentials for a different user account. + // + // Changes made to files in a dev workspace are persistent across API sessions. It's a good + // idea to commit any changes you've made to the git repository, but not strictly required. Your modified files + // reside in a special user-specific directory on the Looker server and will still be there when you login in again + // later and use update_session(workspace_id: "dev") to select the dev workspace for the new API session. + // + rpc Workspace(WorkspaceReq) returns (stream WorkspaceResponse); + + +} \ No newline at end of file diff --git a/proto/grpc_proxy/src/main/resources/simplelogger.properties b/proto/grpc_proxy/src/main/resources/simplelogger.properties new file mode 100644 index 000000000..0ad2bafe4 --- /dev/null +++ b/proto/grpc_proxy/src/main/resources/simplelogger.properties @@ -0,0 +1,2 @@ +org.slf4j.simpleLogger.defaultLogLevel=info +org.slf4j.simpleLogger.log.com.google.looker=debug diff --git a/proto/grpc_proxy/src/test/java/com/google/looker/test/AuthorizationTests.java b/proto/grpc_proxy/src/test/java/com/google/looker/test/AuthorizationTests.java new file mode 100644 index 000000000..be5ce1f08 --- /dev/null +++ b/proto/grpc_proxy/src/test/java/com/google/looker/test/AuthorizationTests.java @@ -0,0 +1,48 @@ +package com.google.looker.test; + +import static org.junit.jupiter.api.Assertions.assertNotNull; +import static org.junit.jupiter.api.Assertions.assertNull; + +import com.google.looker.client.LookerGrpcClient; +import com.google.looker.common.Constants; +import com.google.looker.grpc.services.AccessToken; +import com.google.looker.grpc.services.LoginReq; +import com.google.looker.grpc.services.LoginResponse; +import com.google.looker.grpc.services.LookerServiceGrpc; +import javax.net.ssl.SSLException; +import org.junit.jupiter.api.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +public class AuthorizationTests { + + final private static Logger LOGGER = LoggerFactory.getLogger(AuthorizationTests.class); + + @Test + void rawLogin() throws SSLException { + LOGGER.debug("run login test"); + LookerGrpcClient lookerGrpcClient = new LookerGrpcClient(); + LookerServiceGrpc.LookerServiceBlockingStub stub = lookerGrpcClient.getLookerServiceBlockingStub(); + LoginResponse loginResponse = stub.login( + LoginReq + .newBuilder() + .setClientId(System.getProperty(Constants.LOOKER_CLIENT_ID)) + .setClientSecret(System.getProperty(Constants.LOOKER_CLIENT_SECRET)) + .build() + ); + AccessToken accessToken = loginResponse.getResult(); + assertNotNull(accessToken); + assertNotNull(accessToken.getAccessToken()); + } + + @Test + void clientLogout() throws SSLException { + LOGGER.debug("run login test"); + LookerGrpcClient lookerGrpcClient = new LookerGrpcClient(); + lookerGrpcClient.login(); + assertNotNull(lookerGrpcClient.getAccessToken()); + lookerGrpcClient.logout(); + assertNull(lookerGrpcClient.getAccessToken()); + } + +} diff --git a/proto/grpc_proxy/src/test/java/com/google/looker/test/ConnectionTests.java b/proto/grpc_proxy/src/test/java/com/google/looker/test/ConnectionTests.java new file mode 100644 index 000000000..f3ff47b9d --- /dev/null +++ b/proto/grpc_proxy/src/test/java/com/google/looker/test/ConnectionTests.java @@ -0,0 +1,155 @@ +package com.google.looker.test; + +import static org.junit.jupiter.api.Assertions.assertEquals; +import static org.junit.jupiter.api.Assertions.assertNotNull; +import static org.junit.jupiter.api.Assertions.assertTrue; + +import com.google.looker.client.LookerGrpcClient; +import com.google.looker.grpc.services.AllConnectionsReq; +import com.google.looker.grpc.services.AllConnectionsResponse; +import com.google.looker.grpc.services.ConnectionReq; +import com.google.looker.grpc.services.ConnectionResponse; +import com.google.looker.grpc.services.CreateConnectionReq; +import com.google.looker.grpc.services.CreateConnectionResponse; +import com.google.looker.grpc.services.DBConnection; +import com.google.looker.grpc.services.DeleteConnectionReq; +import com.google.looker.grpc.services.DeleteConnectionResponse; +import com.google.looker.grpc.services.LookerServiceGrpc; +import com.google.looker.grpc.services.TestConnectionReq; +import com.google.looker.grpc.services.TestConnectionResponse; +import com.google.looker.grpc.services.UpdateConnectionReq; +import com.google.looker.grpc.services.UpdateConnectionResponse; +import javax.net.ssl.SSLException; +import org.junit.jupiter.api.AfterEach; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +public class ConnectionTests { + + final private static Logger LOGGER = LoggerFactory.getLogger(ConnectionTests.class); + final private LookerGrpcClient lookerGrpcClient = new LookerGrpcClient(); + private LookerServiceGrpc.LookerServiceBlockingStub stub; + final private static String TEST_CONNECTION_NAME = "testconnection"; + + @BeforeEach + void login() throws SSLException { + lookerGrpcClient.login(); + stub = lookerGrpcClient.getLookerServiceBlockingStub(); + } + + @AfterEach + void logout() throws SSLException { + stub = null; + lookerGrpcClient.logout(); + } + + @Test + void connections() throws SSLException { + LOGGER.debug("run connections test"); + // Get all connections + AllConnectionsResponse allConnectionsResponse = stub + .allConnections( + AllConnectionsReq + .newBuilder() + .build() + ); + assertTrue(allConnectionsResponse.getResultCount() > 0); + + // Get a single connection + String connectionName = allConnectionsResponse.getResult(0).getName(); + ConnectionResponse connectionResponse = stub + .connection( + ConnectionReq + .newBuilder() + .setConnectionName(connectionName) + .setFields("name") + .build() + ); + assertNotNull(connectionResponse.getResult()); + assertEquals(connectionName, connectionResponse.getResult().getName()); + + // Create a connection + cleanupTestConnection(); + CreateConnectionResponse createConnectionResponse = stub.createConnection( + CreateConnectionReq + .newBuilder() + .setBody( + DBConnection + .newBuilder() + .setName(TEST_CONNECTION_NAME) + .setDialectName("mysql") + .setHost("db1.looker.com") + .setPort("3306") + .setUsername(System.getProperty("TEST_LOOKER_USERNAME") + "X") + .setPassword(System.getProperty("TEST_LOOKER_PASSWORD")) + .setDatabase("demo_db2") + .setTmpDbName("looker_demo_scratch") + .build() + ) + .build() + ); + assertNotNull(createConnectionResponse.getResult()); + assertEquals(TEST_CONNECTION_NAME, createConnectionResponse.getResult().getName()); + + // Update connection + UpdateConnectionResponse updateConnectionResponse = stub.updateConnection( + UpdateConnectionReq + .newBuilder() + .setConnectionName(TEST_CONNECTION_NAME) + .setBody( + DBConnection + .newBuilder() + .setUsername(System.getProperty("TEST_LOOKER_USERNAME")) + .build() + ) + .build() + ); + assertNotNull(updateConnectionResponse.getResult()); + assertEquals(System.getProperty("TEST_LOOKER_USERNAME"), + updateConnectionResponse.getResult().getUsername()); + + // Test connection + TestConnectionResponse testConnectionResponse = stub.testConnection(TestConnectionReq + .newBuilder() + .setConnectionName(System.getProperty("TEST_CONNECTION_NAME")) + .setTests("connect") + .build()); + assertTrue(testConnectionResponse.getResultCount() > 0); + assertEquals("Can connect", testConnectionResponse.getResult(0).getMessage()); + + // Delete connection + DeleteConnectionResponse deleteConnectionResponse = stub.deleteConnection( + DeleteConnectionReq + .newBuilder() + .setConnectionName(TEST_CONNECTION_NAME) + .build() + ); + assertNotNull(deleteConnectionResponse.getResult()); + assertEquals("", deleteConnectionResponse.getResult()); + } + + private void cleanupTestConnection() { + try { + stub.connection( + ConnectionReq + .newBuilder() + .setConnectionName(TEST_CONNECTION_NAME) + .setFields("name") + .build() + ); + stub.deleteConnection( + DeleteConnectionReq + .newBuilder() + .setConnectionName(TEST_CONNECTION_NAME) + .build() + ); + } catch(RuntimeException e) { + // noop - not supposed to be there + } + + } + + // +} diff --git a/proto/grpc_proxy/src/test/java/com/google/looker/test/PingTests.java b/proto/grpc_proxy/src/test/java/com/google/looker/test/PingTests.java new file mode 100644 index 000000000..57918bd2d --- /dev/null +++ b/proto/grpc_proxy/src/test/java/com/google/looker/test/PingTests.java @@ -0,0 +1,26 @@ +package com.google.looker.test; + +import static org.junit.jupiter.api.Assertions.assertEquals; + +import com.google.looker.client.LookerGrpcClient; +import com.google.looker.grpc.services.PingServiceGrpc; +import com.google.looker.server.rtl.PingRequest; +import javax.net.ssl.SSLException; +import org.junit.jupiter.api.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +public class PingTests { + + final private static Logger LOGGER = LoggerFactory.getLogger(PingTests.class); + + @Test + void ping() throws SSLException { + LOGGER.debug("run ping test"); + LookerGrpcClient lookerGrpcClient = new LookerGrpcClient(); + PingServiceGrpc.PingServiceBlockingStub stub = lookerGrpcClient.getPingBlockingStub(); + boolean active = stub.ping((PingRequest.newBuilder().build())).getActive(); + assertEquals(true, active); + } + +} diff --git a/proto/grpc_proxy/src/test/java/com/google/looker/test/StreamingUserTests.java b/proto/grpc_proxy/src/test/java/com/google/looker/test/StreamingUserTests.java new file mode 100644 index 000000000..335d49335 --- /dev/null +++ b/proto/grpc_proxy/src/test/java/com/google/looker/test/StreamingUserTests.java @@ -0,0 +1,72 @@ +package com.google.looker.test; + +import static org.junit.jupiter.api.Assertions.assertTrue; + +import com.google.looker.client.LookerGrpcClient; +import com.google.looker.grpc.services.AllUsersReq; +import com.google.looker.grpc.services.AllUsersStreamResponse; +import com.google.looker.grpc.services.LookerStreamingServiceGrpc; +import com.google.looker.grpc.services.User; +import io.grpc.stub.StreamObserver; +import java.util.ArrayList; +import java.util.List; +import java.util.concurrent.CountDownLatch; +import java.util.concurrent.TimeUnit; +import javax.net.ssl.SSLException; +import org.junit.jupiter.api.AfterEach; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +public class StreamingUserTests { + + final private static Logger LOGGER = LoggerFactory.getLogger(StreamingUserTests.class); + final private LookerGrpcClient lookerGrpcClient = new LookerGrpcClient(); + private LookerStreamingServiceGrpc.LookerStreamingServiceStub stub; + + @BeforeEach + void login() throws SSLException { + lookerGrpcClient.login(); + stub = lookerGrpcClient.getLookerStreamingServiceStub(); + } + + @AfterEach + void logout() throws SSLException { + stub = null; + lookerGrpcClient.logout(); + } + + @Test + void allUsers() throws InterruptedException { + LOGGER.debug("run allUsers test"); + CountDownLatch latch = new CountDownLatch(1); + List users = new ArrayList<>(); + int[] countChunks = {0}; + StreamObserver responseObserver = new StreamObserver() { + + @Override + public void onNext(AllUsersStreamResponse value) { + countChunks[0] += 1; + users.add(value.getResult()); + } + + @Override + public void onError(Throwable t) { + } + + @Override + public void onCompleted() { + latch.countDown(); + } + }; + stub.allUsers( + AllUsersReq.newBuilder().build(), + responseObserver + ); + latch.await(3, TimeUnit.SECONDS); + assertTrue(countChunks[0] > 0); + assertTrue(users.size() > 0); + } + +} diff --git a/proto/grpc_proxy/src/test/resources/simplelogger.properties b/proto/grpc_proxy/src/test/resources/simplelogger.properties new file mode 100644 index 000000000..358c38a13 --- /dev/null +++ b/proto/grpc_proxy/src/test/resources/simplelogger.properties @@ -0,0 +1,2 @@ +org.slf4j.simpleLogger.defaultLogLevel=info +org.slf4j.simpleLogger.log.com.google.looker=info diff --git a/spec/Looker.3.1.json b/spec/Looker.3.1.json index e12f249ba..e954aac74 100644 --- a/spec/Looker.3.1.json +++ b/spec/Looker.3.1.json @@ -2,16 +2,16 @@ "swagger": "2.0", "info": { "version": "3.1.0", - "x-looker-release-version": "22.2.3", + "x-looker-release-version": "22.3.0", "title": "Looker API 3.1 Reference", - "description": "### Authorization\n\nThe classic method of API authorization uses Looker **API3** credentials for authorization and access control.\nLooker admins can create API3 credentials on Looker's **Admin/Users** page.\n\nAPI 4.0 adds additional ways to authenticate API requests, including OAuth and CORS requests.\n\nFor details, see [Looker API Authorization](https://looker.com/docs/r/api/authorization).\n\n\n### API Explorer\n\nThe API Explorer is a Looker-provided utility with many new and unique features for learning and using the Looker API and SDKs.\nIt is a replacement for the 'api-docs' page currently provided on Looker instances.\n\nFor details, see the [API Explorer documentation](https://looker.com/docs/r/api/explorer).\n\n\n### Looker Language SDKs\n\nThe Looker API is a RESTful system that should be usable by any programming language capable of making\nHTTPS requests. SDKs for a variety of programming languages are also provided to streamline using the API. Looker\nhas an OpenSource [sdk-codegen project](https://github.com/looker-open-source/sdk-codegen) that provides several\nlanguage SDKs. Language SDKs generated by `sdk-codegen` have an Authentication manager that can automatically\nauthenticate API requests when needed.\n\nFor details on available Looker SDKs, see [Looker API Client SDKs](https://looker.com/docs/r/api/client_sdks).\n\n\n### API Versioning\n\nFuture releases of Looker expand the latest API version release-by-release to securely expose more and more of the core\npower of the Looker platform to API client applications. API endpoints marked as \"beta\" may receive breaking changes without\nwarning (but we will try to avoid doing that). Stable (non-beta) API endpoints should not receive breaking\nchanges in future releases.\n\nFor details, see [Looker API Versioning](https://looker.com/docs/r/api/versioning).\n\n\n### Try It Out!\n\nThis section describes the existing 'api-docs' page available on Looker instances. We recommend using the\n[API Explorer](https://looker.com/docs/r/api/explorer) instead.\n\nThe 'api-docs' page served by the Looker instance includes 'Try It Out!' buttons for each API method. After logging\nin with API3 credentials, you can use the \"Try It Out!\" buttons to call the API directly from the documentation\npage to interactively explore API features and responses.\n\n**NOTE**! With great power comes great responsibility: The \"Try It Out!\" button makes API calls to your live Looker\ninstance. Be especially careful with destructive API operations such as `delete_user` or similar.\nThere is no \"undo\" for API operations. (API Explorer's \"Run It\" feature requires a check mark before running\nAPI operations that can change data.)\n\n\n### In This Release\n\nThe following are a few examples of noteworthy items that have changed between API 3.0 and API 3.1.\nFor more comprehensive coverage of API changes, please see the release notes for your Looker release.\n\n### Examples of new things added in API 3.1 (compared to API 3.0):\n\n* [Dashboard construction](#!/3.1/Dashboard/) APIs\n* [Themes](#!/3.1/Theme/) and [custom color collections](#!/3.1/ColorCollection) APIs\n* Create and run [SQL Runner](#!/3.1/Query/run_sql_query) queries\n* Create and run [merged results](#!/3.1/Query/create_merge_query) queries\n* Create and modify [dashboard filters](#!/3.1/Dashboard/create_dashboard_filter)\n* Create and modify [password requirements](#!/3.1/Auth/password_config)\n\n### Deprecated in API 3.0\n\nThe following functions and properties have been deprecated in API 3.0. They continue to exist and work in API 3.0\nfor the next several Looker releases but they have not been carried forward to API 3.1:\n\n* Dashboard Prefetch functions\n* User access_filter functions\n* User API 1.0 credentials functions\n* Space.is_root and Space.is_user_root properties. Use Space.is_shared_root and Space.is_users_root instead.\n\n### Semantic changes in API 3.1:\n\n* [all_looks()](#!/3.1/Look/all_looks) no longer includes soft-deleted looks, matching [all_dashboards()](#!/3.1/Dashboard/all_dashboards) behavior.\nYou can find soft-deleted looks using [search_looks()](#!/3.1/Look/search_looks) with the `deleted` param set to True.\n* [all_spaces()](#!/3.1/Space/all_spaces) no longer includes duplicate items\n* [search_users()](#!/3.1/User/search_users) no longer accepts Y,y,1,0,N,n for Boolean params, only \"true\" and \"false\".\n* For greater client and network compatibility, [render_task_results](#!/3.1/RenderTask/render_task_results) now returns\nHTTP status **202 Accepted** instead of HTTP status **102 Processing**\n* [all_running_queries()](#!/3.1/Query/all_running_queries) and [kill_query](#!/3.1/Query/kill_query) functions have moved into the [Query](#!/3.1/Query/) function group.\n\nThe API Explorer can be used to [interactively compare](https://looker.com/docs/r/api/explorer#comparing_api_versions) the differences between API 3.1 and 4.0.\n\n\n### API and SDK Support Policies\n\nLooker API versions and language SDKs have varying support levels. Please read the API and SDK\n[support policies](https://looker.com/docs/r/api/support-policy) for more information.\n\n\n", + "description": "### Authorization\n\nThe classic method of API authorization uses Looker **API3** credentials for authorization and access control.\nLooker admins can create API3 credentials on Looker's **Admin/Users** page.\n\nAPI 4.0 adds additional ways to authenticate API requests, including OAuth and CORS requests.\n\nFor details, see [Looker API Authorization](https://looker.com/docs/r/api/authorization).\n\n\n### API Explorer\n\nThe API Explorer is a Looker-provided utility with many new and unique features for learning and using the Looker API and SDKs.\n\nFor details, see the [API Explorer documentation](https://looker.com/docs/r/api/explorer).\n\n\n### Looker Language SDKs\n\nThe Looker API is a RESTful system that should be usable by any programming language capable of making\nHTTPS requests. SDKs for a variety of programming languages are also provided to streamline using the API. Looker\nhas an OpenSource [sdk-codegen project](https://github.com/looker-open-source/sdk-codegen) that provides several\nlanguage SDKs. Language SDKs generated by `sdk-codegen` have an Authentication manager that can automatically\nauthenticate API requests when needed.\n\nFor details on available Looker SDKs, see [Looker API Client SDKs](https://looker.com/docs/r/api/client_sdks).\n\n\n### API Versioning\n\nFuture releases of Looker expand the latest API version release-by-release to securely expose more and more of the core\npower of the Looker platform to API client applications. API endpoints marked as \"beta\" may receive breaking changes without\nwarning (but we will try to avoid doing that). Stable (non-beta) API endpoints should not receive breaking\nchanges in future releases.\n\nFor details, see [Looker API Versioning](https://looker.com/docs/r/api/versioning).\n\n\n### 3.1 Legacy API\n\nAPI 3.1 is now **deprecated**. API 4.0 should be used instead.\n\nThe text below is retained for reference purposes.\n\nThe following are a few examples of noteworthy items that have changed between API 3.0 and API 3.1.\nFor more comprehensive coverage of API changes, please see the release notes for your Looker release.\n\n### Examples of new things added in API 3.1 (compared to API 3.0):\n\n* [Dashboard construction](#!/3.1/Dashboard/) APIs\n* [Themes](#!/3.1/Theme/) and [custom color collections](#!/3.1/ColorCollection) APIs\n* Create and run [SQL Runner](#!/3.1/Query/run_sql_query) queries\n* Create and run [merged results](#!/3.1/Query/create_merge_query) queries\n* Create and modify [dashboard filters](#!/3.1/Dashboard/create_dashboard_filter)\n* Create and modify [password requirements](#!/3.1/Auth/password_config)\n\n### Deprecated in API 3.0\n\nThe following functions and properties have been deprecated in API 3.0. They continue to exist and work in API 3.0\nfor the next several Looker releases but they have not been carried forward to API 3.1:\n\n* Dashboard Prefetch functions\n* User access_filter functions\n* User API 1.0 credentials functions\n* Space.is_root and Space.is_user_root properties. Use Space.is_shared_root and Space.is_users_root instead.\n\n### Semantic changes in API 3.1:\n\n* [all_looks()](#!/3.1/Look/all_looks) no longer includes soft-deleted looks, matching [all_dashboards()](#!/3.1/Dashboard/all_dashboards) behavior.\nYou can find soft-deleted looks using [search_looks()](#!/3.1/Look/search_looks) with the `deleted` param set to True.\n* [all_spaces()](#!/3.1/Space/all_spaces) no longer includes duplicate items\n* [search_users()](#!/3.1/User/search_users) no longer accepts Y,y,1,0,N,n for Boolean params, only \"true\" and \"false\".\n* For greater client and network compatibility, [render_task_results](#!/3.1/RenderTask/render_task_results) now returns\nHTTP status **202 Accepted** instead of HTTP status **102 Processing**\n* [all_running_queries()](#!/3.1/Query/all_running_queries) and [kill_query](#!/3.1/Query/kill_query) functions have moved into the [Query](#!/3.1/Query/) function group.\n\nThe API Explorer can be used to [interactively compare](https://looker.com/docs/r/api/explorer#comparing_api_versions) the differences between API 3.1 and 4.0.\n\n\n### API and SDK Support Policies\n\nLooker API versions and language SDKs have varying support levels. Please read the API and SDK\n[support policies](https://looker.com/docs/r/api/support-policy) for more information.\n\n\n", "contact": { "name": "Looker Team", "url": "https://help.looker.com" }, "license": { "name": "EULA", - "url": "https://localhost:10000/eula" + "url": "https://self-signed.looker.com:9999/eula" } }, "basePath": "/api/3.1", @@ -21,7 +21,7 @@ "produces": [ "application/json" ], - "host": "localhost:20000", + "host": "self-signed.looker.com:19999", "schemes": [ "https" ], @@ -919,7 +919,7 @@ ], "operationId": "run_url_encoded_query", "summary": "Run Url Encoded Query", - "description": "### Run an URL encoded query.\n\nThis requires the caller to encode the specifiers for the query into the URL query part using\nLooker-specific syntax as explained below.\n\nGenerally, you would want to use one of the methods that takes the parameters as json in the POST body\nfor creating and/or running queries. This method exists for cases where one really needs to encode the\nparameters into the URL of a single 'GET' request. This matches the way that the Looker UI formats\n'explore' URLs etc.\n\nThe parameters here are very similar to the json body formatting except that the filter syntax is\ntricky. Unfortunately, this format makes this method not currently callable via the 'Try it out!' button\nin this documentation page. But, this is callable when creating URLs manually or when using the Looker SDK.\n\nHere is an example inline query URL:\n\n```\nhttps://looker.mycompany.com:19999/api/3.0/queries/models/thelook/views/inventory_items/run/json?fields=category.name,inventory_items.days_in_inventory_tier,products.count&f[category.name]=socks&sorts=products.count+desc+0&limit=500&query_timezone=America/Los_Angeles\n```\n\nWhen invoking this endpoint with the Ruby SDK, pass the query parameter parts as a hash. The hash to match the above would look like:\n\n```ruby\nquery_params =\n{\n :fields => \"category.name,inventory_items.days_in_inventory_tier,products.count\",\n :\"f[category.name]\" => \"socks\",\n :sorts => \"products.count desc 0\",\n :limit => \"500\",\n :query_timezone => \"America/Los_Angeles\"\n}\nresponse = ruby_sdk.run_url_encoded_query('thelook','inventory_items','json', query_params)\n\n```\n\nAgain, it is generally easier to use the variant of this method that passes the full query in the POST body.\nThis method is available for cases where other alternatives won't fit the need.\n\nSupported formats:\n\n| result_format | Description\n| :-----------: | :--- |\n| json | Plain json\n| json_detail | Row data plus metadata describing the fields, pivots, table calcs, and other aspects of the query\n| csv | Comma separated values with a header\n| txt | Tab separated values with a header\n| html | Simple html\n| md | Simple markdown\n| xlsx | MS Excel spreadsheet\n| sql | Returns the generated SQL rather than running the query\n| png | A PNG image of the visualization of the query\n| jpg | A JPG image of the visualization of the query\n\n\n", + "description": "### Run an URL encoded query.\n\nThis requires the caller to encode the specifiers for the query into the URL query part using\nLooker-specific syntax as explained below.\n\nGenerally, you would want to use one of the methods that takes the parameters as json in the POST body\nfor creating and/or running queries. This method exists for cases where one really needs to encode the\nparameters into the URL of a single 'GET' request. This matches the way that the Looker UI formats\n'explore' URLs etc.\n\nThe parameters here are very similar to the json body formatting except that the filter syntax is\ntricky. Unfortunately, this format makes this method not currently callable via the 'Try it out!' button\nin this documentation page. But, this is callable when creating URLs manually or when using the Looker SDK.\n\nHere is an example inline query URL:\n\n```\nhttps://looker.mycompany.com:19999/api/3.0/queries/models/thelook/views/inventory_items/run/json?fields=category.name,inventory_items.days_in_inventory_tier,products.count&f[category.name]=socks&sorts=products.count+desc+0&limit=500&query_timezone=America/Los_Angeles\n```\n\nWhen invoking this endpoint with the Ruby SDK, pass the query parameter parts as a hash. The hash to match the above would look like:\n\n```ruby\nquery_params =\n{\n fields: \"category.name,inventory_items.days_in_inventory_tier,products.count\",\n :\"f[category.name]\" => \"socks\",\n sorts: \"products.count desc 0\",\n limit: \"500\",\n query_timezone: \"America/Los_Angeles\"\n}\nresponse = ruby_sdk.run_url_encoded_query('thelook','inventory_items','json', query_params)\n\n```\n\nAgain, it is generally easier to use the variant of this method that passes the full query in the POST body.\nThis method is available for cases where other alternatives won't fit the need.\n\nSupported formats:\n\n| result_format | Description\n| :-----------: | :--- |\n| json | Plain json\n| json_detail | Row data plus metadata describing the fields, pivots, table calcs, and other aspects of the query\n| csv | Comma separated values with a header\n| txt | Tab separated values with a header\n| html | Simple html\n| md | Simple markdown\n| xlsx | MS Excel spreadsheet\n| sql | Returns the generated SQL rather than running the query\n| png | A PNG image of the visualization of the query\n| jpg | A JPG image of the visualization of the query\n\n\n", "produces": [ "text", "application/json", @@ -2850,7 +2850,7 @@ ], "operationId": "create_dashboard", "summary": "Create Dashboard", - "description": "### Create a new dashboard\n\nCreates a new dashboard object and returns the details of the newly created dashboard.\n\n`Title`, `user_id`, and `space_id` are all required fields.\n`Space_id` and `user_id` must contain the id of an existing space or user, respectively.\nA dashboard's `title` must be unique within the space in which it resides.\n\nIf you receive a 422 error response when creating a dashboard, be sure to look at the\nresponse body for information about exactly which fields are missing or contain invalid data.\n\nYou can **update** an existing dashboard with [update_dashboard()](#!/Dashboard/update_dashboard)\n\nYou can **permanently delete** an existing dashboard with [delete_dashboard()](#!/Dashboard/delete_dashboard)\n", + "description": "### Create a new dashboard\n\nCreates a new dashboard object and returns the details of the newly created dashboard.\n\n`Title` and `space_id` are required fields.\n`Space_id` must contain the id of an existing space.\nA dashboard's `title` must be unique within the space in which it resides.\n\nIf you receive a 422 error response when creating a dashboard, be sure to look at the\nresponse body for information about exactly which fields are missing or contain invalid data.\n\nYou can **update** an existing dashboard with [update_dashboard()](#!/Dashboard/update_dashboard)\n\nYou can **permanently delete** an existing dashboard with [delete_dashboard()](#!/Dashboard/delete_dashboard)\n", "parameters": [ { "name": "body", @@ -3023,22 +3023,6 @@ "type": "integer", "format": "int64" }, - { - "name": "limit", - "in": "query", - "description": "Number of results to return. (used with offset and takes priority over page and per_page)", - "required": false, - "type": "integer", - "format": "int64" - }, - { - "name": "offset", - "in": "query", - "description": "Number of results to skip before returning any. (used with limit and takes priority over page and per_page)", - "required": false, - "type": "integer", - "format": "int64" - }, { "name": "sorts", "in": "query", @@ -3812,6 +3796,13 @@ "description": "Requested fields.", "required": false, "type": "string" + }, + { + "name": "apply_filters", + "in": "query", + "description": "Apply relevant filters on dashboard to this tile", + "required": false, + "type": "boolean" } ], "responses": { @@ -6105,7 +6096,7 @@ { "name": "page", "in": "query", - "description": "Requested page.", + "description": "Return only page N of paginated results", "required": false, "type": "integer", "format": "int64" @@ -6113,7 +6104,7 @@ { "name": "per_page", "in": "query", - "description": "Results per page.", + "description": "Return N rows of data per page", "required": false, "type": "integer", "format": "int64" @@ -6659,7 +6650,7 @@ { "name": "page", "in": "query", - "description": "Requested page.", + "description": "Return only page N of paginated results", "required": false, "type": "integer", "format": "int64" @@ -6667,7 +6658,7 @@ { "name": "per_page", "in": "query", - "description": "Results per page.", + "description": "Return N rows of data per page", "required": false, "type": "integer", "format": "int64" @@ -7644,7 +7635,7 @@ { "name": "homepage_item_id", "in": "path", - "description": "Id of homepage_item", + "description": "Id of homepage item", "required": true, "type": "integer", "format": "int64" @@ -7929,7 +7920,7 @@ { "name": "homepage_section_id", "in": "path", - "description": "Id of homepage_section", + "description": "Id of homepage section", "required": true, "type": "integer", "format": "int64" @@ -8133,7 +8124,7 @@ { "name": "integration_hub_id", "in": "path", - "description": "Id of Integration Hub", + "description": "Id of integration_hub", "required": true, "type": "integer", "format": "int64" @@ -8180,7 +8171,7 @@ { "name": "integration_hub_id", "in": "path", - "description": "Id of Integration Hub", + "description": "Id of integration_hub", "required": true, "type": "integer", "format": "int64" @@ -9433,22 +9424,6 @@ "type": "integer", "format": "int64" }, - { - "name": "limit", - "in": "query", - "description": "Number of results to return. (used with offset and takes priority over page and per_page)", - "required": false, - "type": "integer", - "format": "int64" - }, - { - "name": "offset", - "in": "query", - "description": "Number of results to skip before returning any. (used with limit and takes priority over page and per_page)", - "required": false, - "type": "integer", - "format": "int64" - }, { "name": "sorts", "in": "query", @@ -10405,7 +10380,7 @@ ], "responses": { "204": { - "description": "Model set succssfully deleted.", + "description": "Model set successfully deleted.", "schema": { "type": "string" } @@ -11159,7 +11134,7 @@ { "name": "permission_set_id", "in": "path", - "description": "id of permission set", + "description": "Id of permission set", "required": true, "type": "integer", "format": "int64" @@ -13648,7 +13623,7 @@ { "name": "role_id", "in": "path", - "description": "Id of Role", + "description": "id of role", "required": true, "type": "integer", "format": "int64" @@ -13661,8 +13636,7 @@ "schema": { "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" } } } @@ -13718,7 +13692,7 @@ { "name": "role_id", "in": "path", - "description": "id of user", + "description": "id of role", "required": true, "type": "integer", "format": "int64" @@ -13788,8 +13762,7 @@ "schema": { "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" } } } @@ -17647,7 +17620,7 @@ { "name": "user_attribute_id", "in": "path", - "description": "Id of user_attribute", + "description": "Id of user attribute", "required": true, "type": "integer", "format": "int64" @@ -18656,7 +18629,7 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "type": "integer", "format": "int64" @@ -18703,7 +18676,7 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "type": "integer", "format": "int64" @@ -18777,7 +18750,7 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "type": "integer", "format": "int64" @@ -18845,7 +18818,7 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "type": "integer", "format": "int64" @@ -18893,7 +18866,7 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "type": "integer", "format": "int64" @@ -18940,7 +18913,7 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "type": "integer", "format": "int64" @@ -19014,7 +18987,7 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "type": "integer", "format": "int64" @@ -19062,7 +19035,7 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "type": "integer", "format": "int64" @@ -19109,7 +19082,7 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "type": "integer", "format": "int64" @@ -19157,7 +19130,7 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "type": "integer", "format": "int64" @@ -19204,7 +19177,7 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "type": "integer", "format": "int64" @@ -19252,7 +19225,7 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "type": "integer", "format": "int64" @@ -19299,7 +19272,7 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "type": "integer", "format": "int64" @@ -19347,7 +19320,7 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "type": "integer", "format": "int64" @@ -19394,7 +19367,7 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "type": "integer", "format": "int64" @@ -19497,7 +19470,7 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "type": "integer", "format": "int64" @@ -19505,7 +19478,7 @@ { "name": "credentials_api3_id", "in": "path", - "description": "id of API 3 Credential", + "description": "Id of API 3 Credential", "required": true, "type": "integer", "format": "int64" @@ -19553,7 +19526,7 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "type": "integer", "format": "int64" @@ -19603,7 +19576,7 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "type": "integer", "format": "int64" @@ -19734,7 +19707,7 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "type": "integer", "format": "int64" @@ -19742,7 +19715,7 @@ { "name": "credentials_embed_id", "in": "path", - "description": "id of Embedding Credential", + "description": "Id of Embedding Credential", "required": true, "type": "integer", "format": "int64" @@ -19790,7 +19763,7 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "type": "integer", "format": "int64" @@ -19842,7 +19815,7 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "type": "integer", "format": "int64" @@ -19889,7 +19862,7 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "type": "integer", "format": "int64" @@ -19992,7 +19965,7 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "type": "integer", "format": "int64" @@ -20000,7 +19973,7 @@ { "name": "session_id", "in": "path", - "description": "id of Web Login Session", + "description": "Id of Web Login Session", "required": true, "type": "integer", "format": "int64" @@ -20048,7 +20021,7 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "type": "integer", "format": "int64" @@ -20156,7 +20129,7 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "type": "integer", "format": "int64" @@ -20213,7 +20186,7 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "type": "integer", "format": "int64" @@ -20226,8 +20199,7 @@ "schema": { "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" } } }, @@ -20795,7 +20767,7 @@ "type": "string", "readOnly": true, "description": "Timezone in which the Dashboard will run by default.", - "x-looker-nullable": true + "x-looker-nullable": false }, "readonly": { "type": "boolean", @@ -21172,7 +21144,8 @@ "x-looker-nullable": true }, "id": { - "type": "string", + "type": "integer", + "format": "int64", "readOnly": true, "description": "Unique Id", "x-looker-nullable": false @@ -21260,7 +21233,8 @@ "x-looker-nullable": true }, "id": { - "type": "string", + "type": "integer", + "format": "int64", "readOnly": true, "description": "Unique Id", "x-looker-nullable": false @@ -21300,8 +21274,7 @@ "visible_item_order": { "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" }, "readOnly": true, "description": "ids of the homepage items the user can see in the order they should be displayed", @@ -21394,12 +21367,14 @@ "x-looker-nullable": true }, "homepage_section_id": { - "type": "string", + "type": "integer", + "format": "int64", "description": "Associated Homepage Section", "x-looker-nullable": true }, "id": { - "type": "string", + "type": "integer", + "format": "int64", "readOnly": true, "description": "Unique Id", "x-looker-nullable": false @@ -22244,6 +22219,11 @@ "type": "string", "description": "Type", "x-looker-nullable": true + }, + "rich_content_json": { + "type": "string", + "description": "JSON with all the properties required for rich editor and buttons elements", + "x-looker-nullable": true } }, "x-looker-status": "stable" @@ -23294,6 +23274,11 @@ "description": "Count of Alerts associated to a dashboard element", "x-looker-nullable": true }, + "rich_content_json": { + "type": "string", + "description": "JSON with all the properties required for rich editor and buttons elements", + "x-looker-nullable": true + }, "title_text_as_html": { "type": "string", "readOnly": true, @@ -23731,7 +23716,7 @@ "query_timezone": { "type": "string", "description": "Timezone in which the Dashboard will run by default.", - "x-looker-nullable": true + "x-looker-nullable": false }, "readonly": { "type": "boolean", @@ -23793,7 +23778,7 @@ "background_color": { "type": "string", "description": "Background color", - "x-looker-nullable": true + "x-looker-nullable": false }, "created_at": { "type": "string", @@ -23945,22 +23930,22 @@ "text_tile_text_color": { "type": "string", "description": "Color of text on text tiles", - "x-looker-nullable": true + "x-looker-nullable": false }, "tile_background_color": { "type": "string", "description": "Tile background color", - "x-looker-nullable": true + "x-looker-nullable": false }, "tile_text_color": { "type": "string", "description": "Tile text color", - "x-looker-nullable": true + "x-looker-nullable": false }, "title_color": { "type": "string", "description": "Title color", - "x-looker-nullable": true + "x-looker-nullable": false }, "view_count": { "type": "integer", @@ -24170,7 +24155,8 @@ "x-looker-nullable": true }, "id": { - "type": "string", + "type": "integer", + "format": "int64", "readOnly": true, "description": "Unique ID of the datagroup", "x-looker-nullable": false @@ -24775,6 +24761,12 @@ "description": "Whether the dialect supports query cost estimates", "x-looker-nullable": false }, + "cost_estimate_style": { + "type": "string", + "readOnly": true, + "description": "How the dialect handles cost estimation", + "x-looker-nullable": true + }, "persistent_table_indexes": { "type": "string", "readOnly": true, @@ -25373,7 +25365,7 @@ "format": "int64", "readOnly": true, "description": "UNIX timestamp at which this branch was last committed.", - "x-looker-nullable": true + "x-looker-nullable": false }, "ref": { "type": "string", @@ -25765,9 +25757,10 @@ "x-looker-values": [ "cell", "query", - "dashboard" + "dashboard", + "none" ], - "description": "A list of action types the integration supports. Valid values are: \"cell\", \"query\", \"dashboard\".", + "description": "A list of action types the integration supports. Valid values are: \"cell\", \"query\", \"dashboard\", \"none\".", "x-looker-nullable": false }, "supported_formattings": { @@ -29867,7 +29860,7 @@ "type": "string" }, "description": "Pivots", - "x-looker-nullable": true + "x-looker-nullable": false }, "fill_fields": { "type": "array", @@ -29875,7 +29868,7 @@ "type": "string" }, "description": "Fill Fields", - "x-looker-nullable": true + "x-looker-nullable": false }, "filters": { "type": "object", @@ -29883,12 +29876,12 @@ "type": "string" }, "description": "Filters", - "x-looker-nullable": true + "x-looker-nullable": false }, "filter_expression": { "type": "string", "description": "Filter Expression", - "x-looker-nullable": true + "x-looker-nullable": false }, "sorts": { "type": "array", @@ -29901,12 +29894,12 @@ "limit": { "type": "string", "description": "Limit", - "x-looker-nullable": true + "x-looker-nullable": false }, "column_limit": { "type": "string", "description": "Column Limit", - "x-looker-nullable": true + "x-looker-nullable": false }, "total": { "type": "boolean", @@ -29916,7 +29909,7 @@ "row_total": { "type": "string", "description": "Raw Total", - "x-looker-nullable": true + "x-looker-nullable": false }, "subtotals": { "type": "array", @@ -29924,7 +29917,7 @@ "type": "string" }, "description": "Fields on which to run subtotals", - "x-looker-nullable": true + "x-looker-nullable": false }, "vis_config": { "type": "object", @@ -29947,7 +29940,7 @@ "visible_ui_sections": { "type": "string", "description": "Visible UI Sections", - "x-looker-nullable": true + "x-looker-nullable": false }, "slug": { "type": "string", @@ -29958,7 +29951,7 @@ "dynamic_fields": { "type": "string", "description": "Dynamic Fields", - "x-looker-nullable": true + "x-looker-nullable": false }, "client_id": { "type": "string", @@ -29986,7 +29979,7 @@ "query_timezone": { "type": "string", "description": "Query Timezone", - "x-looker-nullable": true + "x-looker-nullable": false }, "has_table_calculations": { "type": "boolean", @@ -29999,7 +29992,7 @@ "format": "double", "x-looker-deprecated": true, "description": "(DEPRECATED) Runtime (Deprecated)", - "x-looker-nullable": true + "x-looker-nullable": false } }, "x-looker-status": "stable", @@ -32051,7 +32044,7 @@ "hidden_value_domain_whitelist": { "type": "string", "description": "Destinations to which a hidden attribute may be sent. Once set, cannot be edited.", - "x-looker-nullable": true + "x-looker-nullable": false } }, "x-looker-status": "stable" @@ -32181,7 +32174,7 @@ "type": "string", "readOnly": true, "description": "If this user attribute is hidden, whitelist of destinations to which it may be sent.", - "x-looker-nullable": true + "x-looker-nullable": false } }, "x-looker-status": "stable" diff --git a/spec/Looker.3.1.oas.json b/spec/Looker.3.1.oas.json index a4e048d12..dd34005f1 100644 --- a/spec/Looker.3.1.oas.json +++ b/spec/Looker.3.1.oas.json @@ -2,16 +2,16 @@ "openapi": "3.0.0", "info": { "version": "3.1.0", - "x-looker-release-version": "22.2.3", + "x-looker-release-version": "22.3.0", "title": "Looker API 3.1 Reference", - "description": "### Authorization\n\nThe classic method of API authorization uses Looker **API3** credentials for authorization and access control.\nLooker admins can create API3 credentials on Looker's **Admin/Users** page.\n\nAPI 4.0 adds additional ways to authenticate API requests, including OAuth and CORS requests.\n\nFor details, see [Looker API Authorization](https://looker.com/docs/r/api/authorization).\n\n\n### API Explorer\n\nThe API Explorer is a Looker-provided utility with many new and unique features for learning and using the Looker API and SDKs.\nIt is a replacement for the 'api-docs' page currently provided on Looker instances.\n\nFor details, see the [API Explorer documentation](https://looker.com/docs/r/api/explorer).\n\n\n### Looker Language SDKs\n\nThe Looker API is a RESTful system that should be usable by any programming language capable of making\nHTTPS requests. SDKs for a variety of programming languages are also provided to streamline using the API. Looker\nhas an OpenSource [sdk-codegen project](https://github.com/looker-open-source/sdk-codegen) that provides several\nlanguage SDKs. Language SDKs generated by `sdk-codegen` have an Authentication manager that can automatically\nauthenticate API requests when needed.\n\nFor details on available Looker SDKs, see [Looker API Client SDKs](https://looker.com/docs/r/api/client_sdks).\n\n\n### API Versioning\n\nFuture releases of Looker expand the latest API version release-by-release to securely expose more and more of the core\npower of the Looker platform to API client applications. API endpoints marked as \"beta\" may receive breaking changes without\nwarning (but we will try to avoid doing that). Stable (non-beta) API endpoints should not receive breaking\nchanges in future releases.\n\nFor details, see [Looker API Versioning](https://looker.com/docs/r/api/versioning).\n\n\n### Try It Out!\n\nThis section describes the existing 'api-docs' page available on Looker instances. We recommend using the\n[API Explorer](https://looker.com/docs/r/api/explorer) instead.\n\nThe 'api-docs' page served by the Looker instance includes 'Try It Out!' buttons for each API method. After logging\nin with API3 credentials, you can use the \"Try It Out!\" buttons to call the API directly from the documentation\npage to interactively explore API features and responses.\n\n**NOTE**! With great power comes great responsibility: The \"Try It Out!\" button makes API calls to your live Looker\ninstance. Be especially careful with destructive API operations such as `delete_user` or similar.\nThere is no \"undo\" for API operations. (API Explorer's \"Run It\" feature requires a check mark before running\nAPI operations that can change data.)\n\n\n### In This Release\n\nThe following are a few examples of noteworthy items that have changed between API 3.0 and API 3.1.\nFor more comprehensive coverage of API changes, please see the release notes for your Looker release.\n\n### Examples of new things added in API 3.1 (compared to API 3.0):\n\n* [Dashboard construction](#!/3.1/Dashboard/) APIs\n* [Themes](#!/3.1/Theme/) and [custom color collections](#!/3.1/ColorCollection) APIs\n* Create and run [SQL Runner](#!/3.1/Query/run_sql_query) queries\n* Create and run [merged results](#!/3.1/Query/create_merge_query) queries\n* Create and modify [dashboard filters](#!/3.1/Dashboard/create_dashboard_filter)\n* Create and modify [password requirements](#!/3.1/Auth/password_config)\n\n### Deprecated in API 3.0\n\nThe following functions and properties have been deprecated in API 3.0. They continue to exist and work in API 3.0\nfor the next several Looker releases but they have not been carried forward to API 3.1:\n\n* Dashboard Prefetch functions\n* User access_filter functions\n* User API 1.0 credentials functions\n* Space.is_root and Space.is_user_root properties. Use Space.is_shared_root and Space.is_users_root instead.\n\n### Semantic changes in API 3.1:\n\n* [all_looks()](#!/3.1/Look/all_looks) no longer includes soft-deleted looks, matching [all_dashboards()](#!/3.1/Dashboard/all_dashboards) behavior.\nYou can find soft-deleted looks using [search_looks()](#!/3.1/Look/search_looks) with the `deleted` param set to True.\n* [all_spaces()](#!/3.1/Space/all_spaces) no longer includes duplicate items\n* [search_users()](#!/3.1/User/search_users) no longer accepts Y,y,1,0,N,n for Boolean params, only \"true\" and \"false\".\n* For greater client and network compatibility, [render_task_results](#!/3.1/RenderTask/render_task_results) now returns\nHTTP status **202 Accepted** instead of HTTP status **102 Processing**\n* [all_running_queries()](#!/3.1/Query/all_running_queries) and [kill_query](#!/3.1/Query/kill_query) functions have moved into the [Query](#!/3.1/Query/) function group.\n\nThe API Explorer can be used to [interactively compare](https://looker.com/docs/r/api/explorer#comparing_api_versions) the differences between API 3.1 and 4.0.\n\n\n### API and SDK Support Policies\n\nLooker API versions and language SDKs have varying support levels. Please read the API and SDK\n[support policies](https://looker.com/docs/r/api/support-policy) for more information.\n\n\n", + "description": "### Authorization\n\nThe classic method of API authorization uses Looker **API3** credentials for authorization and access control.\nLooker admins can create API3 credentials on Looker's **Admin/Users** page.\n\nAPI 4.0 adds additional ways to authenticate API requests, including OAuth and CORS requests.\n\nFor details, see [Looker API Authorization](https://looker.com/docs/r/api/authorization).\n\n\n### API Explorer\n\nThe API Explorer is a Looker-provided utility with many new and unique features for learning and using the Looker API and SDKs.\n\nFor details, see the [API Explorer documentation](https://looker.com/docs/r/api/explorer).\n\n\n### Looker Language SDKs\n\nThe Looker API is a RESTful system that should be usable by any programming language capable of making\nHTTPS requests. SDKs for a variety of programming languages are also provided to streamline using the API. Looker\nhas an OpenSource [sdk-codegen project](https://github.com/looker-open-source/sdk-codegen) that provides several\nlanguage SDKs. Language SDKs generated by `sdk-codegen` have an Authentication manager that can automatically\nauthenticate API requests when needed.\n\nFor details on available Looker SDKs, see [Looker API Client SDKs](https://looker.com/docs/r/api/client_sdks).\n\n\n### API Versioning\n\nFuture releases of Looker expand the latest API version release-by-release to securely expose more and more of the core\npower of the Looker platform to API client applications. API endpoints marked as \"beta\" may receive breaking changes without\nwarning (but we will try to avoid doing that). Stable (non-beta) API endpoints should not receive breaking\nchanges in future releases.\n\nFor details, see [Looker API Versioning](https://looker.com/docs/r/api/versioning).\n\n\n### 3.1 Legacy API\n\nAPI 3.1 is now **deprecated**. API 4.0 should be used instead.\n\nThe text below is retained for reference purposes.\n\nThe following are a few examples of noteworthy items that have changed between API 3.0 and API 3.1.\nFor more comprehensive coverage of API changes, please see the release notes for your Looker release.\n\n### Examples of new things added in API 3.1 (compared to API 3.0):\n\n* [Dashboard construction](#!/3.1/Dashboard/) APIs\n* [Themes](#!/3.1/Theme/) and [custom color collections](#!/3.1/ColorCollection) APIs\n* Create and run [SQL Runner](#!/3.1/Query/run_sql_query) queries\n* Create and run [merged results](#!/3.1/Query/create_merge_query) queries\n* Create and modify [dashboard filters](#!/3.1/Dashboard/create_dashboard_filter)\n* Create and modify [password requirements](#!/3.1/Auth/password_config)\n\n### Deprecated in API 3.0\n\nThe following functions and properties have been deprecated in API 3.0. They continue to exist and work in API 3.0\nfor the next several Looker releases but they have not been carried forward to API 3.1:\n\n* Dashboard Prefetch functions\n* User access_filter functions\n* User API 1.0 credentials functions\n* Space.is_root and Space.is_user_root properties. Use Space.is_shared_root and Space.is_users_root instead.\n\n### Semantic changes in API 3.1:\n\n* [all_looks()](#!/3.1/Look/all_looks) no longer includes soft-deleted looks, matching [all_dashboards()](#!/3.1/Dashboard/all_dashboards) behavior.\nYou can find soft-deleted looks using [search_looks()](#!/3.1/Look/search_looks) with the `deleted` param set to True.\n* [all_spaces()](#!/3.1/Space/all_spaces) no longer includes duplicate items\n* [search_users()](#!/3.1/User/search_users) no longer accepts Y,y,1,0,N,n for Boolean params, only \"true\" and \"false\".\n* For greater client and network compatibility, [render_task_results](#!/3.1/RenderTask/render_task_results) now returns\nHTTP status **202 Accepted** instead of HTTP status **102 Processing**\n* [all_running_queries()](#!/3.1/Query/all_running_queries) and [kill_query](#!/3.1/Query/kill_query) functions have moved into the [Query](#!/3.1/Query/) function group.\n\nThe API Explorer can be used to [interactively compare](https://looker.com/docs/r/api/explorer#comparing_api_versions) the differences between API 3.1 and 4.0.\n\n\n### API and SDK Support Policies\n\nLooker API versions and language SDKs have varying support levels. Please read the API and SDK\n[support policies](https://looker.com/docs/r/api/support-policy) for more information.\n\n\n", "contact": { "name": "Looker Team", "url": "https://help.looker.com" }, "license": { "name": "EULA", - "url": "https://localhost:10000/eula" + "url": "https://self-signed.looker.com:9999/eula" } }, "tags": [ @@ -1319,7 +1319,7 @@ ], "operationId": "run_url_encoded_query", "summary": "Run Url Encoded Query", - "description": "### Run an URL encoded query.\n\nThis requires the caller to encode the specifiers for the query into the URL query part using\nLooker-specific syntax as explained below.\n\nGenerally, you would want to use one of the methods that takes the parameters as json in the POST body\nfor creating and/or running queries. This method exists for cases where one really needs to encode the\nparameters into the URL of a single 'GET' request. This matches the way that the Looker UI formats\n'explore' URLs etc.\n\nThe parameters here are very similar to the json body formatting except that the filter syntax is\ntricky. Unfortunately, this format makes this method not currently callable via the 'Try it out!' button\nin this documentation page. But, this is callable when creating URLs manually or when using the Looker SDK.\n\nHere is an example inline query URL:\n\n```\nhttps://looker.mycompany.com:19999/api/3.0/queries/models/thelook/views/inventory_items/run/json?fields=category.name,inventory_items.days_in_inventory_tier,products.count&f[category.name]=socks&sorts=products.count+desc+0&limit=500&query_timezone=America/Los_Angeles\n```\n\nWhen invoking this endpoint with the Ruby SDK, pass the query parameter parts as a hash. The hash to match the above would look like:\n\n```ruby\nquery_params =\n{\n :fields => \"category.name,inventory_items.days_in_inventory_tier,products.count\",\n :\"f[category.name]\" => \"socks\",\n :sorts => \"products.count desc 0\",\n :limit => \"500\",\n :query_timezone => \"America/Los_Angeles\"\n}\nresponse = ruby_sdk.run_url_encoded_query('thelook','inventory_items','json', query_params)\n\n```\n\nAgain, it is generally easier to use the variant of this method that passes the full query in the POST body.\nThis method is available for cases where other alternatives won't fit the need.\n\nSupported formats:\n\n| result_format | Description\n| :-----------: | :--- |\n| json | Plain json\n| json_detail | Row data plus metadata describing the fields, pivots, table calcs, and other aspects of the query\n| csv | Comma separated values with a header\n| txt | Tab separated values with a header\n| html | Simple html\n| md | Simple markdown\n| xlsx | MS Excel spreadsheet\n| sql | Returns the generated SQL rather than running the query\n| png | A PNG image of the visualization of the query\n| jpg | A JPG image of the visualization of the query\n\n\n", + "description": "### Run an URL encoded query.\n\nThis requires the caller to encode the specifiers for the query into the URL query part using\nLooker-specific syntax as explained below.\n\nGenerally, you would want to use one of the methods that takes the parameters as json in the POST body\nfor creating and/or running queries. This method exists for cases where one really needs to encode the\nparameters into the URL of a single 'GET' request. This matches the way that the Looker UI formats\n'explore' URLs etc.\n\nThe parameters here are very similar to the json body formatting except that the filter syntax is\ntricky. Unfortunately, this format makes this method not currently callable via the 'Try it out!' button\nin this documentation page. But, this is callable when creating URLs manually or when using the Looker SDK.\n\nHere is an example inline query URL:\n\n```\nhttps://looker.mycompany.com:19999/api/3.0/queries/models/thelook/views/inventory_items/run/json?fields=category.name,inventory_items.days_in_inventory_tier,products.count&f[category.name]=socks&sorts=products.count+desc+0&limit=500&query_timezone=America/Los_Angeles\n```\n\nWhen invoking this endpoint with the Ruby SDK, pass the query parameter parts as a hash. The hash to match the above would look like:\n\n```ruby\nquery_params =\n{\n fields: \"category.name,inventory_items.days_in_inventory_tier,products.count\",\n :\"f[category.name]\" => \"socks\",\n sorts: \"products.count desc 0\",\n limit: \"500\",\n query_timezone: \"America/Los_Angeles\"\n}\nresponse = ruby_sdk.run_url_encoded_query('thelook','inventory_items','json', query_params)\n\n```\n\nAgain, it is generally easier to use the variant of this method that passes the full query in the POST body.\nThis method is available for cases where other alternatives won't fit the need.\n\nSupported formats:\n\n| result_format | Description\n| :-----------: | :--- |\n| json | Plain json\n| json_detail | Row data plus metadata describing the fields, pivots, table calcs, and other aspects of the query\n| csv | Comma separated values with a header\n| txt | Tab separated values with a header\n| html | Simple html\n| md | Simple markdown\n| xlsx | MS Excel spreadsheet\n| sql | Returns the generated SQL rather than running the query\n| png | A PNG image of the visualization of the query\n| jpg | A JPG image of the visualization of the query\n\n\n", "parameters": [ { "name": "model_name", @@ -4009,7 +4009,7 @@ ], "operationId": "create_dashboard", "summary": "Create Dashboard", - "description": "### Create a new dashboard\n\nCreates a new dashboard object and returns the details of the newly created dashboard.\n\n`Title`, `user_id`, and `space_id` are all required fields.\n`Space_id` and `user_id` must contain the id of an existing space or user, respectively.\nA dashboard's `title` must be unique within the space in which it resides.\n\nIf you receive a 422 error response when creating a dashboard, be sure to look at the\nresponse body for information about exactly which fields are missing or contain invalid data.\n\nYou can **update** an existing dashboard with [update_dashboard()](#!/Dashboard/update_dashboard)\n\nYou can **permanently delete** an existing dashboard with [delete_dashboard()](#!/Dashboard/delete_dashboard)\n", + "description": "### Create a new dashboard\n\nCreates a new dashboard object and returns the details of the newly created dashboard.\n\n`Title` and `space_id` are required fields.\n`Space_id` must contain the id of an existing space.\nA dashboard's `title` must be unique within the space in which it resides.\n\nIf you receive a 422 error response when creating a dashboard, be sure to look at the\nresponse body for information about exactly which fields are missing or contain invalid data.\n\nYou can **update** an existing dashboard with [update_dashboard()](#!/Dashboard/update_dashboard)\n\nYou can **permanently delete** an existing dashboard with [delete_dashboard()](#!/Dashboard/delete_dashboard)\n", "responses": { "200": { "description": "Dashboard", @@ -4236,26 +4236,6 @@ "format": "int64" } }, - { - "name": "limit", - "in": "query", - "description": "Number of results to return. (used with offset and takes priority over page and per_page)", - "required": false, - "schema": { - "type": "integer", - "format": "int64" - } - }, - { - "name": "offset", - "in": "query", - "description": "Number of results to skip before returning any. (used with limit and takes priority over page and per_page)", - "required": false, - "schema": { - "type": "integer", - "format": "int64" - } - }, { "name": "sorts", "in": "query", @@ -5296,6 +5276,15 @@ "schema": { "type": "string" } + }, + { + "name": "apply_filters", + "in": "query", + "description": "Apply relevant filters on dashboard to this tile", + "required": false, + "schema": { + "type": "boolean" + } } ], "responses": { @@ -8436,7 +8425,7 @@ { "name": "page", "in": "query", - "description": "Requested page.", + "description": "Return only page N of paginated results", "required": false, "schema": { "type": "integer", @@ -8446,7 +8435,7 @@ { "name": "per_page", "in": "query", - "description": "Results per page.", + "description": "Return N rows of data per page", "required": false, "schema": { "type": "integer", @@ -9179,7 +9168,7 @@ { "name": "page", "in": "query", - "description": "Requested page.", + "description": "Return only page N of paginated results", "required": false, "schema": { "type": "integer", @@ -9189,7 +9178,7 @@ { "name": "per_page", "in": "query", - "description": "Results per page.", + "description": "Return N rows of data per page", "required": false, "schema": { "type": "integer", @@ -10496,7 +10485,7 @@ { "name": "homepage_item_id", "in": "path", - "description": "Id of homepage_item", + "description": "Id of homepage item", "required": true, "schema": { "type": "integer", @@ -10885,7 +10874,7 @@ { "name": "homepage_section_id", "in": "path", - "description": "Id of homepage_section", + "description": "Id of homepage section", "required": true, "schema": { "type": "integer", @@ -11163,7 +11152,7 @@ { "name": "integration_hub_id", "in": "path", - "description": "Id of Integration Hub", + "description": "Id of integration_hub", "required": true, "schema": { "type": "integer", @@ -11226,7 +11215,7 @@ { "name": "integration_hub_id", "in": "path", - "description": "Id of Integration Hub", + "description": "Id of integration_hub", "required": true, "schema": { "type": "integer", @@ -12919,26 +12908,6 @@ "format": "int64" } }, - { - "name": "limit", - "in": "query", - "description": "Number of results to return. (used with offset and takes priority over page and per_page)", - "required": false, - "schema": { - "type": "integer", - "format": "int64" - } - }, - { - "name": "offset", - "in": "query", - "description": "Number of results to skip before returning any. (used with limit and takes priority over page and per_page)", - "required": false, - "schema": { - "type": "integer", - "format": "int64" - } - }, { "name": "sorts", "in": "query", @@ -14294,7 +14263,7 @@ ], "responses": { "204": { - "description": "Model set succssfully deleted.", + "description": "Model set successfully deleted.", "content": { "application/json": { "schema": { @@ -15314,7 +15283,7 @@ { "name": "permission_set_id", "in": "path", - "description": "id of permission set", + "description": "Id of permission set", "required": true, "schema": { "type": "integer", @@ -18725,7 +18694,7 @@ { "name": "role_id", "in": "path", - "description": "Id of Role", + "description": "id of role", "required": true, "schema": { "type": "integer", @@ -18796,8 +18765,7 @@ "schema": { "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" } } } @@ -18819,7 +18787,7 @@ { "name": "role_id", "in": "path", - "description": "id of user", + "description": "id of role", "required": true, "schema": { "type": "integer", @@ -18985,8 +18953,7 @@ "schema": { "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" } } } @@ -24157,7 +24124,7 @@ { "name": "user_attribute_id", "in": "path", - "description": "Id of user_attribute", + "description": "Id of user attribute", "required": true, "schema": { "type": "integer", @@ -25495,7 +25462,7 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "schema": { "type": "integer", @@ -25558,7 +25525,7 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "schema": { "type": "integer", @@ -25662,7 +25629,7 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "schema": { "type": "integer", @@ -25756,7 +25723,7 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "schema": { "type": "integer", @@ -25822,7 +25789,7 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "schema": { "type": "integer", @@ -25885,7 +25852,7 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "schema": { "type": "integer", @@ -25989,7 +25956,7 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "schema": { "type": "integer", @@ -26055,7 +26022,7 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "schema": { "type": "integer", @@ -26118,7 +26085,7 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "schema": { "type": "integer", @@ -26184,7 +26151,7 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "schema": { "type": "integer", @@ -26247,7 +26214,7 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "schema": { "type": "integer", @@ -26313,7 +26280,7 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "schema": { "type": "integer", @@ -26376,7 +26343,7 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "schema": { "type": "integer", @@ -26442,7 +26409,7 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "schema": { "type": "integer", @@ -26505,7 +26472,7 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "schema": { "type": "integer", @@ -26644,7 +26611,7 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "schema": { "type": "integer", @@ -26654,7 +26621,7 @@ { "name": "credentials_api3_id", "in": "path", - "description": "id of API 3 Credential", + "description": "Id of API 3 Credential", "required": true, "schema": { "type": "integer", @@ -26720,7 +26687,7 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "schema": { "type": "integer", @@ -26786,7 +26753,7 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "schema": { "type": "integer", @@ -26965,7 +26932,7 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "schema": { "type": "integer", @@ -26975,7 +26942,7 @@ { "name": "credentials_embed_id", "in": "path", - "description": "id of Embedding Credential", + "description": "Id of Embedding Credential", "required": true, "schema": { "type": "integer", @@ -27041,7 +27008,7 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "schema": { "type": "integer", @@ -27109,7 +27076,7 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "schema": { "type": "integer", @@ -27172,7 +27139,7 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "schema": { "type": "integer", @@ -27311,7 +27278,7 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "schema": { "type": "integer", @@ -27321,7 +27288,7 @@ { "name": "session_id", "in": "path", - "description": "id of Web Login Session", + "description": "Id of Web Login Session", "required": true, "schema": { "type": "integer", @@ -27387,7 +27354,7 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "schema": { "type": "integer", @@ -27529,7 +27496,7 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "schema": { "type": "integer", @@ -27604,7 +27571,7 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "schema": { "type": "integer", @@ -27664,8 +27631,7 @@ "schema": { "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" } } } @@ -28272,7 +28238,7 @@ }, "servers": [ { - "url": "https://localhost:20000/api/3.1" + "url": "https://self-signed.looker.com:19999/api/3.1" } ], "components": { @@ -28350,7 +28316,7 @@ "type": "string", "readOnly": true, "description": "Timezone in which the Dashboard will run by default.", - "nullable": true + "nullable": false }, "readonly": { "type": "boolean", @@ -28721,7 +28687,8 @@ "nullable": true }, "id": { - "type": "string", + "type": "integer", + "format": "int64", "readOnly": true, "description": "Unique Id", "nullable": false @@ -28809,7 +28776,8 @@ "nullable": true }, "id": { - "type": "string", + "type": "integer", + "format": "int64", "readOnly": true, "description": "Unique Id", "nullable": false @@ -28849,8 +28817,7 @@ "visible_item_order": { "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" }, "readOnly": true, "description": "ids of the homepage items the user can see in the order they should be displayed", @@ -28943,12 +28910,14 @@ "nullable": true }, "homepage_section_id": { - "type": "string", + "type": "integer", + "format": "int64", "description": "Associated Homepage Section", "nullable": true }, "id": { - "type": "string", + "type": "integer", + "format": "int64", "readOnly": true, "description": "Unique Id", "nullable": false @@ -29759,6 +29728,11 @@ "type": "string", "description": "Type", "nullable": true + }, + "rich_content_json": { + "type": "string", + "description": "JSON with all the properties required for rich editor and buttons elements", + "nullable": true } }, "x-looker-status": "stable" @@ -30798,6 +30772,11 @@ "description": "Count of Alerts associated to a dashboard element", "nullable": true }, + "rich_content_json": { + "type": "string", + "description": "JSON with all the properties required for rich editor and buttons elements", + "nullable": true + }, "title_text_as_html": { "type": "string", "readOnly": true, @@ -31232,7 +31211,7 @@ "query_timezone": { "type": "string", "description": "Timezone in which the Dashboard will run by default.", - "nullable": true + "nullable": false }, "readonly": { "type": "boolean", @@ -31288,7 +31267,7 @@ "background_color": { "type": "string", "description": "Background color", - "nullable": true + "nullable": false }, "created_at": { "type": "string", @@ -31440,22 +31419,22 @@ "text_tile_text_color": { "type": "string", "description": "Color of text on text tiles", - "nullable": true + "nullable": false }, "tile_background_color": { "type": "string", "description": "Tile background color", - "nullable": true + "nullable": false }, "tile_text_color": { "type": "string", "description": "Tile text color", - "nullable": true + "nullable": false }, "title_color": { "type": "string", "description": "Title color", - "nullable": true + "nullable": false }, "view_count": { "type": "integer", @@ -31657,7 +31636,8 @@ "nullable": true }, "id": { - "type": "string", + "type": "integer", + "format": "int64", "readOnly": true, "description": "Unique ID of the datagroup", "nullable": false @@ -32251,6 +32231,12 @@ "description": "Whether the dialect supports query cost estimates", "nullable": false }, + "cost_estimate_style": { + "type": "string", + "readOnly": true, + "description": "How the dialect handles cost estimation", + "nullable": true + }, "persistent_table_indexes": { "type": "string", "readOnly": true, @@ -32849,7 +32835,7 @@ "format": "int64", "readOnly": true, "description": "UNIX timestamp at which this branch was last committed.", - "nullable": true + "nullable": false }, "ref": { "type": "string", @@ -33241,9 +33227,10 @@ "enum": [ "cell", "query", - "dashboard" + "dashboard", + "none" ], - "description": "A list of action types the integration supports. Valid values are: \"cell\", \"query\", \"dashboard\".", + "description": "A list of action types the integration supports. Valid values are: \"cell\", \"query\", \"dashboard\", \"none\".", "nullable": false }, "supported_formattings": { @@ -37280,7 +37267,7 @@ "type": "string" }, "description": "Pivots", - "nullable": true + "nullable": false }, "fill_fields": { "type": "array", @@ -37288,7 +37275,7 @@ "type": "string" }, "description": "Fill Fields", - "nullable": true + "nullable": false }, "filters": { "type": "object", @@ -37296,12 +37283,12 @@ "type": "string" }, "description": "Filters", - "nullable": true + "nullable": false }, "filter_expression": { "type": "string", "description": "Filter Expression", - "nullable": true + "nullable": false }, "sorts": { "type": "array", @@ -37314,12 +37301,12 @@ "limit": { "type": "string", "description": "Limit", - "nullable": true + "nullable": false }, "column_limit": { "type": "string", "description": "Column Limit", - "nullable": true + "nullable": false }, "total": { "type": "boolean", @@ -37329,7 +37316,7 @@ "row_total": { "type": "string", "description": "Raw Total", - "nullable": true + "nullable": false }, "subtotals": { "type": "array", @@ -37337,7 +37324,7 @@ "type": "string" }, "description": "Fields on which to run subtotals", - "nullable": true + "nullable": false }, "vis_config": { "type": "object", @@ -37360,7 +37347,7 @@ "visible_ui_sections": { "type": "string", "description": "Visible UI Sections", - "nullable": true + "nullable": false }, "slug": { "type": "string", @@ -37371,7 +37358,7 @@ "dynamic_fields": { "type": "string", "description": "Dynamic Fields", - "nullable": true + "nullable": false }, "client_id": { "type": "string", @@ -37399,7 +37386,7 @@ "query_timezone": { "type": "string", "description": "Query Timezone", - "nullable": true + "nullable": false }, "has_table_calculations": { "type": "boolean", @@ -37412,7 +37399,7 @@ "format": "double", "x-looker-deprecated": true, "description": "(DEPRECATED) Runtime (Deprecated)", - "nullable": true + "nullable": false } }, "x-looker-status": "stable", @@ -39429,7 +39416,7 @@ "hidden_value_domain_whitelist": { "type": "string", "description": "Destinations to which a hidden attribute may be sent. Once set, cannot be edited.", - "nullable": true + "nullable": false } }, "x-looker-status": "stable" @@ -39559,7 +39546,7 @@ "type": "string", "readOnly": true, "description": "If this user attribute is hidden, whitelist of destinations to which it may be sent.", - "nullable": true + "nullable": false } }, "x-looker-status": "stable" diff --git a/spec/Looker.4.0.json b/spec/Looker.4.0.json index 54dd95e95..df9b09db4 100644 --- a/spec/Looker.4.0.json +++ b/spec/Looker.4.0.json @@ -1,17 +1,17 @@ { "swagger": "2.0", "info": { - "version": "4.0.22.2", - "x-looker-release-version": "22.2.3", - "title": "Looker API 4.0 (Beta) Reference", - "description": "\nWelcome to the future! API 4.0 co-exists with APIs 3.1 and 3.0. (3.0 should no longer be used.)\nThe \"beta\" tag means updates for API 4.0 may include breaking changes, but as always we will work to minimize them.\n\n### Authorization\n\nThe classic method of API authorization uses Looker **API3** credentials for authorization and access control.\nLooker admins can create API3 credentials on Looker's **Admin/Users** page.\n\nAPI 4.0 adds additional ways to authenticate API requests, including OAuth and CORS requests.\n\nFor details, see [Looker API Authorization](https://looker.com/docs/r/api/authorization).\n\n\n### API Explorer\n\nThe API Explorer is a Looker-provided utility with many new and unique features for learning and using the Looker API and SDKs.\nIt is a replacement for the 'api-docs' page currently provided on Looker instances.\n\nFor details, see the [API Explorer documentation](https://looker.com/docs/r/api/explorer).\n\n\n### Looker Language SDKs\n\nThe Looker API is a RESTful system that should be usable by any programming language capable of making\nHTTPS requests. SDKs for a variety of programming languages are also provided to streamline using the API. Looker\nhas an OpenSource [sdk-codegen project](https://github.com/looker-open-source/sdk-codegen) that provides several\nlanguage SDKs. Language SDKs generated by `sdk-codegen` have an Authentication manager that can automatically\nauthenticate API requests when needed.\n\nFor details on available Looker SDKs, see [Looker API Client SDKs](https://looker.com/docs/r/api/client_sdks).\n\n\n### API Versioning\n\nFuture releases of Looker expand the latest API version release-by-release to securely expose more and more of the core\npower of the Looker platform to API client applications. API endpoints marked as \"beta\" may receive breaking changes without\nwarning (but we will try to avoid doing that). Stable (non-beta) API endpoints should not receive breaking\nchanges in future releases.\n\nFor details, see [Looker API Versioning](https://looker.com/docs/r/api/versioning).\n\n\n### In This Release\n\nAPI 4.0 version was introduced so we can make adjustments to API functions, parameters, and response types to\nfix bugs and inconsistencies. These changes fall outside the bounds of non-breaking additive changes we can\nmake to our stable API 3.1.\n\nOne benefit of these type adjustments in API 4.0 is dramatically better support for strongly\ntyped languages like TypeScript, Kotlin, Swift, Go, C#, and more.\n\nWhile API 3.1 is still the de-facto Looker API (\"current\", \"stable\", \"default\", etc), the bulk\nof our development activity has shifted to API 4.0, where all new features are added.\n\nThe API Explorer can be used to [interactively compare](https://looker.com/docs/r/api/explorer#comparing_api_versions) the differences between API 3.1 and 4.0.\n\n\n### API and SDK Support Policies\n\nLooker API versions and language SDKs have varying support levels. Please read the API and SDK\n[support policies](https://looker.com/docs/r/api/support-policy) for more information.\n\n\n", + "version": "4.0.22.3", + "x-looker-release-version": "22.3.0", + "title": "Looker API 4.0 Reference", + "description": "\nAPI 4.0 is the current release of the Looker API. API 3.1 is deprecated.\n\n### Authorization\n\nThe classic method of API authorization uses Looker **API3** credentials for authorization and access control.\nLooker admins can create API3 credentials on Looker's **Admin/Users** page.\n\nAPI 4.0 adds additional ways to authenticate API requests, including OAuth and CORS requests.\n\nFor details, see [Looker API Authorization](https://looker.com/docs/r/api/authorization).\n\n\n### API Explorer\n\nThe API Explorer is a Looker-provided utility with many new and unique features for learning and using the Looker API and SDKs.\n\nFor details, see the [API Explorer documentation](https://looker.com/docs/r/api/explorer).\n\n\n### Looker Language SDKs\n\nThe Looker API is a RESTful system that should be usable by any programming language capable of making\nHTTPS requests. SDKs for a variety of programming languages are also provided to streamline using the API. Looker\nhas an OpenSource [sdk-codegen project](https://github.com/looker-open-source/sdk-codegen) that provides several\nlanguage SDKs. Language SDKs generated by `sdk-codegen` have an Authentication manager that can automatically\nauthenticate API requests when needed.\n\nFor details on available Looker SDKs, see [Looker API Client SDKs](https://looker.com/docs/r/api/client_sdks).\n\n\n### API Versioning\n\nFuture releases of Looker expand the latest API version release-by-release to securely expose more and more of the core\npower of the Looker platform to API client applications. API endpoints marked as \"beta\" may receive breaking changes without\nwarning (but we will try to avoid doing that). Stable (non-beta) API endpoints should not receive breaking\nchanges in future releases.\n\nFor details, see [Looker API Versioning](https://looker.com/docs/r/api/versioning).\n\n\n### In This Release\n\nAPI 4.0 version was introduced to make adjustments to API functions, parameters, and response types to\nfix bugs and inconsistencies. These changes fall outside the bounds of non-breaking additive changes we can\nmake to the previous API 3.1.\n\nOne benefit of these type adjustments in API 4.0 is dramatically better support for strongly\ntyped languages like TypeScript, Kotlin, Swift, Go, C#, and more.\n\nSee the [API 4.0 GA announcement](https://developers.looker.com/api/advanced-usage/version-4-ga) for more information\nabout API 4.0.\n\nThe API Explorer can be used to [interactively compare](https://looker.com/docs/r/api/explorer#comparing_api_versions) the differences between API 3.1 and 4.0.\n\n\n### API and SDK Support Policies\n\nLooker API versions and language SDKs have varying support levels. Please read the API and SDK\n[support policies](https://looker.com/docs/r/api/support-policy) for more information.\n\n\n", "contact": { "name": "Looker Team", "url": "https://help.looker.com" }, "license": { "name": "EULA", - "url": "https://localhost:10000/eula" + "url": "https://self-signed.looker.com:9999/eula" } }, "basePath": "/api/4.0", @@ -21,7 +21,7 @@ "produces": [ "application/json" ], - "host": "localhost:20000", + "host": "self-signed.looker.com:19999", "schemes": [ "https" ], @@ -46,10 +46,6 @@ "name": "ColorCollection", "description": "Manage Color Collections" }, - { - "name": "Command", - "description": "Manage Commands" - }, { "name": "Config", "description": "Manage General Configuration" @@ -465,8 +461,7 @@ "in": "path", "description": "Id of query", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "fields", @@ -636,8 +631,7 @@ "in": "path", "description": "Id of query", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "result_format", @@ -939,7 +933,7 @@ ], "operationId": "run_url_encoded_query", "summary": "Run Url Encoded Query", - "description": "### Run an URL encoded query.\n\nThis requires the caller to encode the specifiers for the query into the URL query part using\nLooker-specific syntax as explained below.\n\nGenerally, you would want to use one of the methods that takes the parameters as json in the POST body\nfor creating and/or running queries. This method exists for cases where one really needs to encode the\nparameters into the URL of a single 'GET' request. This matches the way that the Looker UI formats\n'explore' URLs etc.\n\nThe parameters here are very similar to the json body formatting except that the filter syntax is\ntricky. Unfortunately, this format makes this method not currently callable via the 'Try it out!' button\nin this documentation page. But, this is callable when creating URLs manually or when using the Looker SDK.\n\nHere is an example inline query URL:\n\n```\nhttps://looker.mycompany.com:19999/api/3.0/queries/models/thelook/views/inventory_items/run/json?fields=category.name,inventory_items.days_in_inventory_tier,products.count&f[category.name]=socks&sorts=products.count+desc+0&limit=500&query_timezone=America/Los_Angeles\n```\n\nWhen invoking this endpoint with the Ruby SDK, pass the query parameter parts as a hash. The hash to match the above would look like:\n\n```ruby\nquery_params =\n{\n :fields => \"category.name,inventory_items.days_in_inventory_tier,products.count\",\n :\"f[category.name]\" => \"socks\",\n :sorts => \"products.count desc 0\",\n :limit => \"500\",\n :query_timezone => \"America/Los_Angeles\"\n}\nresponse = ruby_sdk.run_url_encoded_query('thelook','inventory_items','json', query_params)\n\n```\n\nAgain, it is generally easier to use the variant of this method that passes the full query in the POST body.\nThis method is available for cases where other alternatives won't fit the need.\n\nSupported formats:\n\n| result_format | Description\n| :-----------: | :--- |\n| json | Plain json\n| json_detail | Row data plus metadata describing the fields, pivots, table calcs, and other aspects of the query\n| csv | Comma separated values with a header\n| txt | Tab separated values with a header\n| html | Simple html\n| md | Simple markdown\n| xlsx | MS Excel spreadsheet\n| sql | Returns the generated SQL rather than running the query\n| png | A PNG image of the visualization of the query\n| jpg | A JPG image of the visualization of the query\n\n\n", + "description": "### Run an URL encoded query.\n\nThis requires the caller to encode the specifiers for the query into the URL query part using\nLooker-specific syntax as explained below.\n\nGenerally, you would want to use one of the methods that takes the parameters as json in the POST body\nfor creating and/or running queries. This method exists for cases where one really needs to encode the\nparameters into the URL of a single 'GET' request. This matches the way that the Looker UI formats\n'explore' URLs etc.\n\nThe parameters here are very similar to the json body formatting except that the filter syntax is\ntricky. Unfortunately, this format makes this method not currently callable via the 'Try it out!' button\nin this documentation page. But, this is callable when creating URLs manually or when using the Looker SDK.\n\nHere is an example inline query URL:\n\n```\nhttps://looker.mycompany.com:19999/api/3.0/queries/models/thelook/views/inventory_items/run/json?fields=category.name,inventory_items.days_in_inventory_tier,products.count&f[category.name]=socks&sorts=products.count+desc+0&limit=500&query_timezone=America/Los_Angeles\n```\n\nWhen invoking this endpoint with the Ruby SDK, pass the query parameter parts as a hash. The hash to match the above would look like:\n\n```ruby\nquery_params =\n{\n fields: \"category.name,inventory_items.days_in_inventory_tier,products.count\",\n :\"f[category.name]\" => \"socks\",\n sorts: \"products.count desc 0\",\n limit: \"500\",\n query_timezone: \"America/Los_Angeles\"\n}\nresponse = ruby_sdk.run_url_encoded_query('thelook','inventory_items','json', query_params)\n\n```\n\nAgain, it is generally easier to use the variant of this method that passes the full query in the POST body.\nThis method is available for cases where other alternatives won't fit the need.\n\nSupported formats:\n\n| result_format | Description\n| :-----------: | :--- |\n| json | Plain json\n| json_detail | Row data plus metadata describing the fields, pivots, table calcs, and other aspects of the query\n| csv | Comma separated values with a header\n| txt | Tab separated values with a header\n| html | Simple html\n| md | Simple markdown\n| xlsx | MS Excel spreadsheet\n| sql | Returns the generated SQL rather than running the query\n| png | A PNG image of the visualization of the query\n| jpg | A JPG image of the visualization of the query\n\n\n", "produces": [ "text", "application/json", @@ -1070,8 +1064,7 @@ "in": "path", "description": "Id of user.", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "associative", @@ -1266,8 +1259,7 @@ "in": "path", "description": "ID of an alert", "required": true, - "type": "integer", - "format": "int64" + "type": "string" } ], "responses": { @@ -1306,8 +1298,7 @@ "in": "path", "description": "ID of an alert", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "body", @@ -1373,8 +1364,7 @@ "in": "path", "description": "ID of an alert", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "body", @@ -1434,8 +1424,7 @@ "in": "path", "description": "ID of an alert", "required": true, - "type": "integer", - "format": "int64" + "type": "string" } ], "responses": { @@ -1552,8 +1541,7 @@ "in": "path", "description": "ID of an alert", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "force", @@ -2124,242 +2112,6 @@ "x-looker-activity-type": "non_query" } }, - "/commands": { - "post": { - "tags": [ - "Command" - ], - "operationId": "create_command", - "summary": "Create a custom command", - "description": "### Create a new command.\n# Required fields: [:name, :linked_content_id, :linked_content_type]\n# `linked_content_type` must be one of [\"dashboard\", \"lookml_dashboard\"]\n#\n", - "parameters": [ - { - "name": "body", - "in": "body", - "description": "Writable command parameters", - "required": true, - "schema": { - "$ref": "#/definitions/Command" - } - } - ], - "responses": { - "200": { - "description": "The command is saved.", - "schema": { - "$ref": "#/definitions/Command" - } - }, - "400": { - "description": "Bad Request", - "schema": { - "$ref": "#/definitions/Error" - } - }, - "404": { - "description": "Not Found", - "schema": { - "$ref": "#/definitions/Error" - } - }, - "409": { - "description": "Resource Already Exists", - "schema": { - "$ref": "#/definitions/Error" - } - }, - "422": { - "description": "Validation Error", - "schema": { - "$ref": "#/definitions/ValidationError" - } - }, - "429": { - "description": "Too Many Requests", - "schema": { - "$ref": "#/definitions/Error" - } - } - }, - "x-looker-status": "beta", - "x-looker-activity-type": "non_query" - }, - "get": { - "tags": [ - "Command" - ], - "operationId": "get_all_commands", - "summary": "Get All Commands", - "description": "### Get All Commands.\n", - "parameters": [ - { - "name": "content_id", - "in": "query", - "description": "Id of the associated content. This must be accompanied with content_type.", - "required": false, - "type": "string" - }, - { - "name": "content_type", - "in": "query", - "description": "Type of the associated content. This must be accompanied with content_id.", - "required": false, - "type": "string" - }, - { - "name": "limit", - "in": "query", - "description": "Number of results to return.", - "required": false, - "type": "integer", - "format": "int64" - } - ], - "responses": { - "200": { - "description": "Commands", - "schema": { - "type": "array", - "items": { - "$ref": "#/definitions/Command" - } - } - }, - "400": { - "description": "Bad Request", - "schema": { - "$ref": "#/definitions/Error" - } - }, - "404": { - "description": "Not Found", - "schema": { - "$ref": "#/definitions/Error" - } - }, - "405": { - "description": "Resource Can't Be Modified", - "schema": { - "$ref": "#/definitions/Error" - } - } - }, - "x-looker-status": "beta", - "x-looker-activity-type": "non_query" - } - }, - "/commands/{command_id}": { - "patch": { - "tags": [ - "Command" - ], - "operationId": "update_command", - "summary": "Update a custom command", - "description": "### Update an existing custom command.\n# Optional fields: ['name', 'description']\n#\n", - "parameters": [ - { - "name": "command_id", - "in": "path", - "description": "ID of a command", - "required": true, - "type": "integer", - "format": "int64" - }, - { - "name": "body", - "in": "body", - "description": "Re-writable command parameters", - "required": true, - "schema": { - "$ref": "#/definitions/UpdateCommand" - } - } - ], - "responses": { - "200": { - "description": "The command is updated.", - "schema": { - "$ref": "#/definitions/Command" - } - }, - "400": { - "description": "Bad Request", - "schema": { - "$ref": "#/definitions/Error" - } - }, - "404": { - "description": "Not Found", - "schema": { - "$ref": "#/definitions/Error" - } - }, - "422": { - "description": "Validation Error", - "schema": { - "$ref": "#/definitions/ValidationError" - } - }, - "429": { - "description": "Too Many Requests", - "schema": { - "$ref": "#/definitions/Error" - } - } - }, - "x-looker-status": "beta", - "x-looker-activity-type": "non_query" - }, - "delete": { - "tags": [ - "Command" - ], - "operationId": "delete_command", - "summary": "Delete a custom command", - "description": "### Delete an existing custom command.\n", - "parameters": [ - { - "name": "command_id", - "in": "path", - "description": "ID of a command", - "required": true, - "type": "integer", - "format": "int64" - } - ], - "responses": { - "204": { - "description": "The command is deleted." - }, - "400": { - "description": "Bad Request", - "schema": { - "$ref": "#/definitions/Error" - } - }, - "404": { - "description": "Not Found", - "schema": { - "$ref": "#/definitions/Error" - } - }, - "405": { - "description": "Resource Can't Be Modified", - "schema": { - "$ref": "#/definitions/Error" - } - }, - "429": { - "description": "Too Many Requests", - "schema": { - "$ref": "#/definitions/Error" - } - } - }, - "x-looker-status": "beta", - "x-looker-activity-type": "non_query" - } - }, "/content_favorite/search": { "get": { "tags": [ @@ -2374,8 +2126,7 @@ "in": "query", "description": "Match content favorite id(s)", "required": false, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "user_id", @@ -2491,8 +2242,7 @@ "in": "path", "description": "Id of favorite content", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "fields", @@ -2538,8 +2288,7 @@ "in": "path", "description": "Id of favorite content", "required": true, - "type": "integer", - "format": "int64" + "type": "string" } ], "responses": { @@ -2647,8 +2396,7 @@ "in": "query", "description": "Parent space of content.", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "fields", @@ -2699,8 +2447,7 @@ "in": "path", "description": "Id of content metadata", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "body", @@ -2760,8 +2507,7 @@ "in": "path", "description": "Id of content metadata", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "fields", @@ -2876,8 +2622,7 @@ "in": "query", "description": "Id of content metadata", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "fields", @@ -2988,8 +2733,7 @@ "in": "path", "description": "Id of content metadata access", "required": true, - "type": "integer", - "format": "int64" + "type": "string" } ], "responses": { @@ -3331,8 +3075,7 @@ "in": "query", "description": "Match credentials_email id.", "required": false, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "email", @@ -3581,7 +3324,7 @@ ], "operationId": "create_dashboard", "summary": "Create Dashboard", - "description": "### Create a new dashboard\n\nCreates a new dashboard object and returns the details of the newly created dashboard.\n\n`Title`, `user_id`, and `space_id` are all required fields.\n`Space_id` and `user_id` must contain the id of an existing space or user, respectively.\nA dashboard's `title` must be unique within the space in which it resides.\n\nIf you receive a 422 error response when creating a dashboard, be sure to look at the\nresponse body for information about exactly which fields are missing or contain invalid data.\n\nYou can **update** an existing dashboard with [update_dashboard()](#!/Dashboard/update_dashboard)\n\nYou can **permanently delete** an existing dashboard with [delete_dashboard()](#!/Dashboard/delete_dashboard)\n", + "description": "### Create a new dashboard\n\nCreates a new dashboard object and returns the details of the newly created dashboard.\n\n`Title` and `space_id` are required fields.\n`Space_id` must contain the id of an existing space.\nA dashboard's `title` must be unique within the space in which it resides.\n\nIf you receive a 422 error response when creating a dashboard, be sure to look at the\nresponse body for information about exactly which fields are missing or contain invalid data.\n\nYou can **update** an existing dashboard with [update_dashboard()](#!/Dashboard/update_dashboard)\n\nYou can **permanently delete** an existing dashboard with [delete_dashboard()](#!/Dashboard/delete_dashboard)\n", "parameters": [ { "name": "body", @@ -3738,18 +3481,20 @@ { "name": "page", "in": "query", - "description": "Requested page.", + "description": "DEPRECATED. Use limit and offset instead. Return only page N of paginated results", "required": false, "type": "integer", - "format": "int64" + "format": "int64", + "x-looker-deprecated": true }, { "name": "per_page", "in": "query", - "description": "Results per page.", + "description": "DEPRECATED. Use limit and offset instead. Return N rows of data per page", "required": false, "type": "integer", - "format": "int64" + "format": "int64", + "x-looker-deprecated": true }, { "name": "limit", @@ -4436,16 +4181,14 @@ "in": "query", "description": "Select elements that refer to a given dashboard id", "required": false, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "look_id", "in": "query", "description": "Select elements that refer to a given look id", "required": false, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "title", @@ -4745,6 +4488,13 @@ "description": "Requested fields.", "required": false, "type": "string" + }, + { + "name": "apply_filters", + "in": "query", + "description": "Apply relevant filters on dashboard to this tile", + "required": false, + "type": "boolean" } ], "responses": { @@ -5663,8 +5413,7 @@ "in": "path", "description": "ID of datagroup.", "required": true, - "type": "integer", - "format": "int64" + "type": "string" } ], "responses": { @@ -5703,8 +5452,7 @@ "in": "path", "description": "ID of datagroup.", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "body", @@ -6819,8 +6567,7 @@ "in": "path", "description": "Id of Embed Secret", "required": true, - "type": "integer", - "format": "int64" + "type": "string" } ], "responses": { @@ -7476,15 +7223,33 @@ { "name": "page", "in": "query", - "description": "Requested page.", + "description": "DEPRECATED. Use limit and offset instead. Return only page N of paginated results", "required": false, "type": "integer", - "format": "int64" + "format": "int64", + "x-looker-deprecated": true }, { "name": "per_page", "in": "query", - "description": "Results per page.", + "description": "DEPRECATED. Use limit and offset instead. Return N rows of data per page", + "required": false, + "type": "integer", + "format": "int64", + "x-looker-deprecated": true + }, + { + "name": "limit", + "in": "query", + "description": "Number of results to return. (used with offset and takes priority over page and per_page)", + "required": false, + "type": "integer", + "format": "int64" + }, + { + "name": "offset", + "in": "query", + "description": "Number of results to skip before returning any. (used with limit and takes priority over page and per_page)", "required": false, "type": "integer", "format": "int64" @@ -7503,8 +7268,7 @@ "required": false, "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" }, "collectionFormat": "csv" }, @@ -7513,8 +7277,7 @@ "in": "query", "description": "Id of content metadata to which groups must have access.", "required": false, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "can_add_to_content_metadata", @@ -7668,8 +7431,7 @@ "in": "query", "description": "Match group id.", "required": false, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "name", @@ -7778,8 +7540,7 @@ "in": "query", "description": "Match group id.", "required": false, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "name", @@ -7888,8 +7649,7 @@ "in": "query", "description": "Match group id.", "required": false, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "name", @@ -7961,8 +7721,7 @@ "in": "path", "description": "Id of group", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "fields", @@ -8008,8 +7767,7 @@ "in": "path", "description": "Id of group", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "body", @@ -8076,8 +7834,7 @@ "in": "path", "description": "Id of group", "required": true, - "type": "integer", - "format": "int64" + "type": "string" } ], "responses": { @@ -8130,8 +7887,7 @@ "in": "path", "description": "Id of group", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "fields", @@ -8180,8 +7936,7 @@ "in": "path", "description": "Id of group", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "body", @@ -8237,8 +7992,7 @@ "in": "path", "description": "Id of group", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "fields", @@ -8250,15 +8004,33 @@ { "name": "page", "in": "query", - "description": "Requested page.", + "description": "DEPRECATED. Use limit and offset instead. Return only page N of paginated results", "required": false, "type": "integer", - "format": "int64" + "format": "int64", + "x-looker-deprecated": true }, { "name": "per_page", "in": "query", - "description": "Results per page.", + "description": "DEPRECATED. Use limit and offset instead. Return N rows of data per page", + "required": false, + "type": "integer", + "format": "int64", + "x-looker-deprecated": true + }, + { + "name": "limit", + "in": "query", + "description": "Number of results to return. (used with offset and takes priority over page and per_page)", + "required": false, + "type": "integer", + "format": "int64" + }, + { + "name": "offset", + "in": "query", + "description": "Number of results to skip before returning any. (used with limit and takes priority over page and per_page)", "required": false, "type": "integer", "format": "int64" @@ -8310,8 +8082,7 @@ "in": "path", "description": "Id of group", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "body", @@ -8367,16 +8138,14 @@ "in": "path", "description": "Id of group", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "user_id", "in": "path", "description": "Id of user to remove from group", "required": true, - "type": "integer", - "format": "int64" + "type": "string" } ], "responses": { @@ -8420,16 +8189,14 @@ "in": "path", "description": "Id of group", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "deleting_group_id", "in": "path", "description": "Id of group to delete", "required": true, - "type": "integer", - "format": "int64" + "type": "string" } ], "responses": { @@ -8473,16 +8240,14 @@ "in": "path", "description": "Id of group", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "user_attribute_id", "in": "path", "description": "Id of user attribute", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "body", @@ -8536,16 +8301,14 @@ "in": "path", "description": "Id of group", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "user_attribute_id", "in": "path", "description": "Id of user attribute", "required": true, - "type": "integer", - "format": "int64" + "type": "string" } ], "responses": { @@ -8747,7 +8510,7 @@ { "name": "page", "in": "query", - "description": "The page to return.", + "description": "The page to return. DEPRECATED. Use offset instead.", "required": false, "type": "integer", "format": "int64" @@ -8755,7 +8518,7 @@ { "name": "per_page", "in": "query", - "description": "The number of items in the returned page.", + "description": "The number of items in the returned page. DEPRECATED. Use limit instead.", "required": false, "type": "integer", "format": "int64" @@ -8825,8 +8588,7 @@ "in": "path", "description": "Id of board", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "fields", @@ -8872,8 +8634,7 @@ "in": "path", "description": "Id of board", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "body", @@ -8940,8 +8701,7 @@ "in": "path", "description": "Id of board", "required": true, - "type": "integer", - "format": "int64" + "type": "string" } ], "responses": { @@ -9112,8 +8872,7 @@ "in": "path", "description": "Id of board item", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "fields", @@ -9159,8 +8918,7 @@ "in": "path", "description": "Id of board item", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "body", @@ -9225,10 +8983,9 @@ { "name": "board_item_id", "in": "path", - "description": "Id of board_item", + "description": "Id of board item", "required": true, - "type": "integer", - "format": "int64" + "type": "string" } ], "responses": { @@ -9436,8 +9193,7 @@ "in": "path", "description": "Id of board section", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "fields", @@ -9483,8 +9239,7 @@ "in": "path", "description": "Id of board section", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "body", @@ -9551,8 +9306,7 @@ "in": "path", "description": "Id of board section", "required": true, - "type": "integer", - "format": "int64" + "type": "string" } ], "responses": { @@ -9708,10 +9462,9 @@ { "name": "integration_hub_id", "in": "path", - "description": "Id of Integration Hub", + "description": "Id of integration_hub", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "fields", @@ -9755,10 +9508,9 @@ { "name": "integration_hub_id", "in": "path", - "description": "Id of Integration Hub", + "description": "Id of integration_hub", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "body", @@ -9826,8 +9578,7 @@ "in": "path", "description": "Id of integration_hub", "required": true, - "type": "integer", - "format": "int64" + "type": "string" } ], "responses": { @@ -9874,8 +9625,7 @@ "in": "path", "description": "Id of integration_hub", "required": true, - "type": "integer", - "format": "int64" + "type": "string" } ], "responses": { @@ -10972,8 +10722,7 @@ "in": "query", "description": "Select looks that reference a particular query by query_id", "required": false, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "curate", @@ -10999,18 +10748,20 @@ { "name": "page", "in": "query", - "description": "Requested page.", + "description": "DEPRECATED. Use limit and offset instead. Return only page N of paginated results", "required": false, "type": "integer", - "format": "int64" + "format": "int64", + "x-looker-deprecated": true }, { "name": "per_page", "in": "query", - "description": "Results per page.", + "description": "DEPRECATED. Use limit and offset instead. Return N rows of data per page", "required": false, "type": "integer", - "format": "int64" + "format": "int64", + "x-looker-deprecated": true }, { "name": "limit", @@ -12680,8 +12431,7 @@ "in": "query", "description": "Match model set id.", "required": false, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "name", @@ -12753,8 +12503,7 @@ "in": "path", "description": "Id of model set", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "fields", @@ -12800,13 +12549,12 @@ "in": "path", "description": "id of model set", "required": true, - "type": "integer", - "format": "int64" + "type": "string" } ], "responses": { "204": { - "description": "Model set succssfully deleted.", + "description": "Model set successfully deleted.", "schema": { "type": "string" } @@ -12846,8 +12594,7 @@ "in": "path", "description": "id of model set", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "body", @@ -13331,8 +13078,7 @@ "in": "path", "description": "The id of the user to enable use of this app", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "fields", @@ -13400,8 +13146,7 @@ "in": "path", "description": "The id of the user to enable use of this app", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "fields", @@ -13845,8 +13590,7 @@ "in": "query", "description": "Match permission set id.", "required": false, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "name", @@ -13918,8 +13662,7 @@ "in": "path", "description": "Id of permission set", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "fields", @@ -13965,8 +13708,7 @@ "in": "path", "description": "Id of permission set", "required": true, - "type": "integer", - "format": "int64" + "type": "string" } ], "responses": { @@ -14015,10 +13757,9 @@ { "name": "permission_set_id", "in": "path", - "description": "id of permission set", + "description": "Id of permission set", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "body", @@ -15410,8 +15151,7 @@ "in": "path", "description": "Id of look to render", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "result_format", @@ -15500,8 +15240,7 @@ "in": "path", "description": "Id of the query to render", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "result_format", @@ -16075,8 +15814,7 @@ "required": false, "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" }, "collectionFormat": "csv" } @@ -16211,8 +15949,7 @@ "in": "query", "description": "Match role id.", "required": false, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "name", @@ -16307,8 +16044,7 @@ "in": "query", "description": "Match role id.", "required": false, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "name", @@ -16373,8 +16109,7 @@ "in": "path", "description": "id of role", "required": true, - "type": "integer", - "format": "int64" + "type": "string" } ], "responses": { @@ -16413,8 +16148,7 @@ "in": "path", "description": "id of role", "required": true, - "type": "integer", - "format": "int64" + "type": "string" } ], "responses": { @@ -16465,8 +16199,7 @@ "in": "path", "description": "id of role", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "body", @@ -16534,8 +16267,7 @@ "in": "path", "description": "id of role", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "fields", @@ -16582,10 +16314,9 @@ { "name": "role_id", "in": "path", - "description": "Id of Role", + "description": "id of role", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "body", @@ -16595,8 +16326,7 @@ "schema": { "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" } } } @@ -16652,10 +16382,9 @@ { "name": "role_id", "in": "path", - "description": "id of user", + "description": "id of role", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "fields", @@ -16711,8 +16440,7 @@ "in": "path", "description": "id of role", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "body", @@ -16722,8 +16450,7 @@ "schema": { "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" } } } @@ -17157,8 +16884,7 @@ "in": "path", "description": "Space Id", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "fields", @@ -17209,8 +16935,7 @@ "in": "path", "description": "Scheduled Plan Id", "required": true, - "type": "integer", - "format": "int64" + "type": "string" } ], "responses": { @@ -17255,8 +16980,7 @@ "in": "path", "description": "Scheduled Plan Id", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "body", @@ -17316,8 +17040,7 @@ "in": "path", "description": "Scheduled Plan Id", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "fields", @@ -17424,8 +17147,7 @@ "in": "query", "description": "Return scheduled plans belonging to this user_id. If not provided, returns scheduled plans owned by the caller.", "required": false, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "fields", @@ -17551,16 +17273,14 @@ "in": "path", "description": "Look Id", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "user_id", "in": "query", "description": "User Id (default is requesting user if not specified)", "required": false, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "fields", @@ -17618,16 +17338,14 @@ "in": "path", "description": "Dashboard Id", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "user_id", "in": "query", "description": "User Id (default is requesting user if not specified)", "required": false, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "all_users", @@ -17692,8 +17410,7 @@ "in": "query", "description": "User Id (default is requesting user if not specified)", "required": false, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "fields", @@ -17751,8 +17468,7 @@ "in": "path", "description": "Id of schedule plan to copy and run", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "body", @@ -17984,7 +17700,7 @@ ], "operationId": "set_setting", "summary": "Set Setting", - "description": "### Configure Looker Settings\n\nAvailable settings are:\n - extension_framework_enabled\n - marketplace_auto_install_enabled\n - marketplace_enabled\n - whitelabel_configuration\n - custom_welcome_email\n - onboarding_enabled\n\nSee the `Setting` type for more information on the specific values that can be configured.\n", + "description": "### Configure Looker Settings\n\nAvailable settings are:\n - extension_framework_enabled\n - marketplace_auto_install_enabled\n - marketplace_enabled\n - privatelabel_configuration\n - custom_welcome_email\n - onboarding_enabled\n\nSee the `Setting` type for more information on the specific values that can be configured.\n", "parameters": [ { "name": "body", @@ -18050,7 +17766,7 @@ ], "operationId": "get_setting", "summary": "Get Setting", - "description": "### Get Looker Settings\n\nAvailable settings are:\n - extension_framework_enabled\n - marketplace_auto_install_enabled\n - marketplace_enabled\n - whitelabel_configuration\n - custom_welcome_email\n - onboarding_enabled\n\n", + "description": "### Get Looker Settings\n\nAvailable settings are:\n - extension_framework_enabled\n - marketplace_auto_install_enabled\n - marketplace_enabled\n - privatelabel_configuration\n - custom_welcome_email\n - onboarding_enabled\n\n", "parameters": [ { "name": "fields", @@ -18269,8 +17985,7 @@ "in": "query", "description": "Match Space id", "required": false, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "parent_id", @@ -19494,8 +19209,7 @@ "in": "query", "description": "Match theme id.", "required": false, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "name", @@ -19867,8 +19581,7 @@ "in": "path", "description": "Id of theme", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "fields", @@ -19914,8 +19627,7 @@ "in": "path", "description": "Id of theme", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "body", @@ -20786,8 +20498,7 @@ "in": "path", "description": "Id of user attribute", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "fields", @@ -20833,8 +20544,7 @@ "in": "path", "description": "Id of user attribute", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "body", @@ -20899,10 +20609,9 @@ { "name": "user_attribute_id", "in": "path", - "description": "Id of user_attribute", + "description": "Id of user attribute", "required": true, - "type": "integer", - "format": "int64" + "type": "string" } ], "responses": { @@ -20949,8 +20658,7 @@ "in": "path", "description": "Id of user attribute", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "fields", @@ -20999,8 +20707,7 @@ "in": "path", "description": "Id of user attribute", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "body", @@ -21123,15 +20830,33 @@ { "name": "page", "in": "query", - "description": "Return only page N of paginated results", + "description": "DEPRECATED. Use limit and offset instead. Return only page N of paginated results", "required": false, "type": "integer", - "format": "int64" + "format": "int64", + "x-looker-deprecated": true }, { "name": "per_page", "in": "query", - "description": "Return N rows of data per page", + "description": "DEPRECATED. Use limit and offset instead. Return N rows of data per page", + "required": false, + "type": "integer", + "format": "int64", + "x-looker-deprecated": true + }, + { + "name": "limit", + "in": "query", + "description": "Number of results to return. (used with offset and takes priority over page and per_page)", + "required": false, + "type": "integer", + "format": "int64" + }, + { + "name": "offset", + "in": "query", + "description": "Number of results to skip before returning any. (used with limit and takes priority over page and per_page)", "required": false, "type": "integer", "format": "int64" @@ -21310,7 +21035,8 @@ "description": "DEPRECATED. Use limit and offset instead. Return only page N of paginated results", "required": false, "type": "integer", - "format": "int64" + "format": "int64", + "x-looker-deprecated": true }, { "name": "per_page", @@ -21318,7 +21044,8 @@ "description": "DEPRECATED. Use limit and offset instead. Return N rows of data per page", "required": false, "type": "integer", - "format": "int64" + "format": "int64", + "x-looker-deprecated": true }, { "name": "limit", @@ -21350,8 +21077,7 @@ "required": false, "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" }, "collectionFormat": "csv" } @@ -21465,7 +21191,8 @@ "description": "DEPRECATED. Use limit and offset instead. Return only page N of paginated results", "required": false, "type": "integer", - "format": "int64" + "format": "int64", + "x-looker-deprecated": true }, { "name": "per_page", @@ -21473,7 +21200,8 @@ "description": "DEPRECATED. Use limit and offset instead. Return N rows of data per page", "required": false, "type": "integer", - "format": "int64" + "format": "int64", + "x-looker-deprecated": true }, { "name": "limit", @@ -21625,7 +21353,8 @@ "description": "DEPRECATED. Use limit and offset instead. Return only page N of paginated results", "required": false, "type": "integer", - "format": "int64" + "format": "int64", + "x-looker-deprecated": true }, { "name": "per_page", @@ -21633,7 +21362,8 @@ "description": "DEPRECATED. Use limit and offset instead. Return N rows of data per page", "required": false, "type": "integer", - "format": "int64" + "format": "int64", + "x-looker-deprecated": true }, { "name": "limit", @@ -21663,8 +21393,7 @@ "in": "query", "description": "Match User Id", "required": false, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "first_name", @@ -21743,8 +21472,7 @@ "in": "path", "description": "Id of user", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "fields", @@ -21790,8 +21518,7 @@ "in": "path", "description": "Id of user", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "body", @@ -21852,8 +21579,7 @@ "in": "path", "description": "Id of user", "required": true, - "type": "integer", - "format": "int64" + "type": "string" } ], "responses": { @@ -21953,10 +21679,9 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "fields", @@ -22000,10 +21725,9 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "body", @@ -22074,10 +21798,9 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "body", @@ -22142,10 +21865,9 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, - "type": "integer", - "format": "int64" + "type": "string" } ], "responses": { @@ -22190,10 +21912,9 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "fields", @@ -22237,10 +21958,9 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "body", @@ -22311,10 +22031,9 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, - "type": "integer", - "format": "int64" + "type": "string" } ], "responses": { @@ -22359,10 +22078,9 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "fields", @@ -22406,10 +22124,9 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, - "type": "integer", - "format": "int64" + "type": "string" } ], "responses": { @@ -22454,10 +22171,9 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "fields", @@ -22501,10 +22217,9 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, - "type": "integer", - "format": "int64" + "type": "string" } ], "responses": { @@ -22549,10 +22264,9 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "fields", @@ -22596,10 +22310,9 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, - "type": "integer", - "format": "int64" + "type": "string" } ], "responses": { @@ -22644,10 +22357,9 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "fields", @@ -22691,10 +22403,9 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, - "type": "integer", - "format": "int64" + "type": "string" } ], "responses": { @@ -22741,16 +22452,14 @@ "in": "path", "description": "Id of user", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "credentials_api3_id", "in": "path", "description": "Id of API 3 Credential", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "fields", @@ -22794,18 +22503,16 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "credentials_api3_id", "in": "path", - "description": "id of API 3 Credential", + "description": "Id of API 3 Credential", "required": true, - "type": "integer", - "format": "int64" + "type": "string" } ], "responses": { @@ -22850,10 +22557,9 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "fields", @@ -22900,10 +22606,9 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "fields", @@ -22969,16 +22674,14 @@ "in": "path", "description": "Id of user", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "credentials_embed_id", "in": "path", "description": "Id of Embedding Credential", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "fields", @@ -23022,18 +22725,16 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "credentials_embed_id", "in": "path", - "description": "id of Embedding Credential", + "description": "Id of Embedding Credential", "required": true, - "type": "integer", - "format": "int64" + "type": "string" } ], "responses": { @@ -23078,10 +22779,9 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "fields", @@ -23130,10 +22830,9 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "fields", @@ -23177,10 +22876,9 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, - "type": "integer", - "format": "int64" + "type": "string" } ], "responses": { @@ -23227,16 +22925,14 @@ "in": "path", "description": "Id of user", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "session_id", "in": "path", "description": "Id of Web Login Session", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "fields", @@ -23280,18 +22976,16 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "session_id", "in": "path", - "description": "id of Web Login Session", + "description": "Id of Web Login Session", "required": true, - "type": "integer", - "format": "int64" + "type": "string" } ], "responses": { @@ -23336,10 +23030,9 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "fields", @@ -23390,8 +23083,7 @@ "in": "path", "description": "Id of user", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "expires", @@ -23444,10 +23136,9 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "fields", @@ -23501,10 +23192,9 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "body", @@ -23514,8 +23204,7 @@ "schema": { "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" } } }, @@ -23568,8 +23257,7 @@ "in": "path", "description": "Id of user", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "fields", @@ -23585,8 +23273,7 @@ "required": false, "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" }, "collectionFormat": "csv" }, @@ -23640,16 +23327,14 @@ "in": "path", "description": "Id of user", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "user_attribute_id", "in": "path", "description": "Id of user attribute", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "body", @@ -23703,16 +23388,14 @@ "in": "path", "description": "Id of user", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "user_attribute_id", "in": "path", "description": "Id of user attribute", "required": true, - "type": "integer", - "format": "int64" + "type": "string" } ], "responses": { @@ -23750,8 +23433,7 @@ "in": "path", "description": "Id of user", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "fields", @@ -23799,8 +23481,7 @@ "in": "path", "description": "Id of user", "required": true, - "type": "integer", - "format": "int64" + "type": "string" }, { "name": "body", @@ -24261,15 +23942,13 @@ "x-looker-nullable": false }, "content_favorite_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Content Favorite Id", "x-looker-nullable": true }, "content_metadata_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of content metadata", "x-looker-nullable": true @@ -24336,8 +24015,7 @@ "x-looker-nullable": true }, "user_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of User", "x-looker-nullable": true @@ -24388,8 +24066,7 @@ "x-looker-nullable": true }, "homepage_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "Id reference to parent homepage", "x-looker-nullable": true }, @@ -24403,8 +24080,7 @@ "x-looker-nullable": true }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "x-looker-nullable": false @@ -24418,8 +24094,7 @@ "item_order": { "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" }, "description": "ids of the homepage items in the order they should be displayed", "x-looker-nullable": true @@ -24444,8 +24119,7 @@ "visible_item_order": { "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" }, "readOnly": true, "description": "ids of the homepage items the user can see in the order they should be displayed", @@ -24472,15 +24146,13 @@ "x-looker-nullable": true }, "content_favorite_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Content favorite id associated with the item this content is based on", "x-looker-nullable": true }, "content_metadata_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Content metadata id associated with the item this content is based on", "x-looker-nullable": true @@ -24519,8 +24191,7 @@ "x-looker-nullable": true }, "dashboard_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "Dashboard to base this item on", "x-looker-nullable": true }, @@ -24538,14 +24209,12 @@ "x-looker-nullable": true }, "homepage_section_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "Associated Homepage Section", "x-looker-nullable": true }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "x-looker-nullable": false @@ -24857,8 +24526,7 @@ "x-looker-nullable": true }, "dashboard_element_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "ID of the dashboard element associated with the alert. Refer to [dashboard_element()](#!/Dashboard/DashboardElement)", "x-looker-nullable": true }, @@ -24893,8 +24561,7 @@ "x-looker-nullable": false }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "ID of the alert", "x-looker-nullable": false @@ -24944,8 +24611,7 @@ "x-looker-nullable": true }, "owner_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "User id of alert owner", "x-looker-nullable": false }, @@ -25013,8 +24679,7 @@ "AlertPatch": { "properties": { "owner_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "New owner ID of the alert", "x-looker-nullable": true }, @@ -25059,8 +24724,7 @@ "x-looker-nullable": true }, "sudo_user_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "The id of the actual user in the case when this session represents one user sudo'ing as another", "x-looker-nullable": true @@ -25134,15 +24798,13 @@ "x-looker-nullable": true }, "content_favorite_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Content favorite id associated with the item this content is based on", "x-looker-nullable": true }, "content_metadata_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Content metadata id associated with the item this content is based on", "x-looker-nullable": true @@ -25169,8 +24831,7 @@ "x-looker-nullable": true }, "dashboard_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "Dashboard to base this item on", "x-looker-nullable": true }, @@ -25188,14 +24849,12 @@ "x-looker-nullable": true }, "board_section_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "Associated Board Section", "x-looker-nullable": true }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "x-looker-nullable": false @@ -25240,12 +24899,44 @@ "description": "Relative url for the associated content", "x-looker-nullable": false }, + "use_custom_description": { + "type": "boolean", + "description": "Whether the custom description should be used instead of the content description, if the item is associated with content", + "x-looker-nullable": false + }, + "use_custom_title": { + "type": "boolean", + "description": "Whether the custom title should be used instead of the content title, if the item is associated with content", + "x-looker-nullable": false + }, + "use_custom_url": { + "type": "boolean", + "description": "Whether the custom url should be used instead of the content url, if the item is associated with content", + "x-looker-nullable": false + }, "view_count": { "type": "integer", "format": "int64", "readOnly": true, "description": "Number of times content has been viewed, if present", "x-looker-nullable": true + }, + "custom_image_data_base64": { + "type": "string", + "x-looker-write-only": true, + "description": "(Write-Only) base64 encoded image data", + "x-looker-nullable": true + }, + "custom_image_url": { + "type": "string", + "readOnly": true, + "description": "Custom image_url entered by the user, if present", + "x-looker-nullable": true + }, + "use_custom_image": { + "type": "boolean", + "description": "Whether the custom image should be used instead of the content image, if the item is associated with content", + "x-looker-nullable": false } }, "x-looker-status": "beta" @@ -25262,8 +24953,7 @@ "x-looker-nullable": false }, "content_metadata_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of associated content_metadata record", "x-looker-nullable": true @@ -25296,8 +24986,7 @@ "x-looker-nullable": true }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "x-looker-nullable": false @@ -25305,8 +24994,7 @@ "section_order": { "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" }, "description": "ids of the board sections in the order they should be displayed", "x-looker-nullable": true @@ -25324,8 +25012,7 @@ "x-looker-nullable": true }, "user_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "User id of board creator", "x-looker-nullable": true @@ -25369,8 +25056,7 @@ "x-looker-nullable": true }, "board_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "Id reference to parent board", "x-looker-nullable": true }, @@ -25384,8 +25070,7 @@ "x-looker-nullable": true }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "x-looker-nullable": false @@ -25393,8 +25078,7 @@ "item_order": { "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" }, "description": "ids of the board items in the order they should be displayed", "x-looker-nullable": true @@ -25402,8 +25086,7 @@ "visible_item_order": { "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" }, "readOnly": true, "description": "ids of the homepage items the user can see in the order they should be displayed", @@ -25538,82 +25221,21 @@ }, "x-looker-status": "stable" }, - "Command": { - "properties": { - "id": { - "type": "integer", - "format": "int64", - "readOnly": true, - "description": "Id of the command record", - "x-looker-nullable": false - }, - "author_id": { - "type": "integer", - "format": "int64", - "readOnly": true, - "description": "Id of the command author", - "x-looker-nullable": false - }, - "name": { - "type": "string", - "description": "Name of the command", - "x-looker-nullable": false - }, - "description": { - "type": "string", - "description": "Description of the command", - "x-looker-nullable": true - }, - "linked_content_id": { - "type": "string", - "description": "Id of the content associated with the command", - "x-looker-nullable": false - }, - "linked_content_type": { - "type": "string", - "x-looker-values": [ - "dashboard", - "lookml_dashboard" - ], - "description": "Name of the command Valid values are: \"dashboard\", \"lookml_dashboard\".", - "x-looker-nullable": false - } - }, - "x-looker-status": "beta" - }, - "UpdateCommand": { - "properties": { - "name": { - "type": "string", - "description": "Name of the command", - "x-looker-nullable": true - }, - "description": { - "type": "string", - "description": "Description of the command", - "x-looker-nullable": true - } - }, - "x-looker-status": "beta" - }, "ContentFavorite": { "properties": { "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "x-looker-nullable": false }, "user_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "User Id which owns this ContentFavorite", "x-looker-nullable": false }, "content_metadata_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "Content Metadata Id associated with this ContentFavorite", "x-looker-nullable": false }, @@ -25624,8 +25246,7 @@ "x-looker-nullable": true }, "dashboard_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of a dashboard", "x-looker-nullable": true @@ -25643,8 +25264,7 @@ "x-looker-nullable": true }, "board_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of a board", "x-looker-nullable": true @@ -25686,15 +25306,13 @@ "x-looker-nullable": true }, "group_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "ID of associated group", "x-looker-nullable": true }, "user_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "ID of associated user", "x-looker-nullable": true @@ -25714,8 +25332,7 @@ "x-looker-nullable": false }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "x-looker-nullable": false @@ -25727,8 +25344,7 @@ "x-looker-nullable": true }, "parent_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of Parent Content", "x-looker-nullable": true @@ -25763,8 +25379,7 @@ "x-looker-nullable": false }, "inheriting_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of Inherited Content", "x-looker-nullable": true @@ -26026,8 +25641,7 @@ "x-looker-nullable": true }, "query_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "Id Of Query", "x-looker-nullable": true }, @@ -26055,6 +25669,11 @@ "type": "string", "description": "Type", "x-looker-nullable": true + }, + "rich_content_json": { + "type": "string", + "description": "JSON with all the properties required for rich editor and buttons elements", + "x-looker-nullable": true } }, "x-looker-status": "stable" @@ -26159,8 +25778,7 @@ "x-looker-nullable": true }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "x-looker-nullable": false @@ -26171,8 +25789,7 @@ "ContentValidationAlert": { "properties": { "id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "ID of the alert", "x-looker-nullable": false }, @@ -26246,8 +25863,7 @@ "x-looker-nullable": false }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "x-looker-nullable": false @@ -26259,8 +25875,7 @@ "x-looker-nullable": true }, "dashboard_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of the viewed Dashboard", "x-looker-nullable": true @@ -26272,22 +25887,19 @@ "x-looker-nullable": true }, "content_metadata_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Content metadata id of the Look or Dashboard", "x-looker-nullable": true }, "user_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of user content was viewed by", "x-looker-nullable": true }, "group_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of group content was viewed by", "x-looker-nullable": true @@ -26373,15 +25985,13 @@ "CreateOAuthApplicationUserStateResponse": { "properties": { "user_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "User Id", "x-looker-nullable": false }, "oauth_application_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "OAuth Application ID", "x-looker-nullable": false @@ -26405,8 +26015,7 @@ "x-looker-nullable": false }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "x-looker-nullable": false @@ -26457,8 +26066,7 @@ "x-looker-nullable": false }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "x-looker-nullable": false @@ -26669,8 +26277,7 @@ "x-looker-nullable": true }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "x-looker-nullable": false @@ -27226,8 +26833,7 @@ "x-looker-nullable": true }, "query_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "Id Of Query", "x-looker-nullable": true }, @@ -27249,8 +26855,7 @@ "x-looker-nullable": true }, "result_maker_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "ID of the ResultMakerLookup entry.", "x-looker-nullable": true }, @@ -27286,6 +26891,11 @@ "description": "Count of Alerts associated to a dashboard element", "x-looker-nullable": true }, + "rich_content_json": { + "type": "string", + "description": "JSON with all the properties required for rich editor and buttons elements", + "x-looker-nullable": true + }, "title_text_as_html": { "type": "string", "readOnly": true, @@ -27685,15 +27295,13 @@ "x-looker-nullable": false }, "content_favorite_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Content Favorite Id", "x-looker-nullable": true }, "content_metadata_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of content metadata", "x-looker-nullable": true @@ -27755,8 +27363,7 @@ "x-looker-nullable": true }, "user_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of User", "x-looker-nullable": true @@ -27833,8 +27440,7 @@ "x-looker-nullable": true }, "deleter_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of User that 'soft' deleted the dashboard.", "x-looker-nullable": true @@ -27880,8 +27486,7 @@ "x-looker-nullable": true }, "last_updater_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of User that most recently updated the dashboard.", "x-looker-nullable": true @@ -27911,12 +27516,12 @@ "show_filters_bar": { "type": "boolean", "description": "Show filters bar. **Security Note:** This property only affects the *cosmetic* appearance of the dashboard, not a user's ability to access data. Hiding the filters bar does **NOT** prevent users from changing filters by other means. For information on how to set up secure data access control policies, see [Control User Access to Data](https://looker.com/docs/r/api/control-access)", - "x-looker-nullable": false + "x-looker-nullable": true }, "show_title": { "type": "boolean", "description": "Show title", - "x-looker-nullable": false + "x-looker-nullable": true }, "folder_id": { "type": "string", @@ -28157,8 +27762,7 @@ "x-looker-nullable": true }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique ID of the datagroup", "x-looker-nullable": false @@ -28480,8 +28084,7 @@ "x-looker-nullable": true }, "oauth_application_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "An External OAuth Application to use for authenticating to the database", "x-looker-nullable": true }, @@ -28625,8 +28228,7 @@ "x-looker-nullable": false }, "installation_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Installation ID", "x-looker-nullable": false @@ -28792,6 +28394,12 @@ "description": "Whether the dialect supports query cost estimates", "x-looker-nullable": false }, + "cost_estimate_style": { + "type": "string", + "readOnly": true, + "description": "How the dialect handles cost estimation", + "x-looker-nullable": true + }, "persistent_table_indexes": { "type": "string", "readOnly": true, @@ -28975,8 +28583,7 @@ "group_ids": { "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" }, "description": "List of Looker group ids in which to enroll the embed user", "x-looker-nullable": true @@ -28996,8 +28603,7 @@ "x-looker-nullable": true }, "secret_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "Id of the embed secret to use to sign this SSO url. If specified, the value must be an id of a valid (active) secret defined in the Looker instance. If not specified, the URL will be signed with the newest active embed secret defined in the Looker instance.", "x-looker-nullable": true } @@ -29026,8 +28632,7 @@ "x-looker-nullable": false }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "x-looker-nullable": false @@ -29039,8 +28644,7 @@ "x-looker-nullable": true }, "user_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of user who created this secret", "x-looker-nullable": true @@ -29117,8 +28721,7 @@ "x-looker-nullable": false }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "ID of this OAuth Application", "x-looker-nullable": false @@ -29173,8 +28776,7 @@ "x-looker-nullable": false }, "content_metadata_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of content metadata", "x-looker-nullable": true @@ -29187,8 +28789,7 @@ "x-looker-nullable": true }, "creator_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "User Id of Creator", "x-looker-nullable": true @@ -29316,8 +28917,7 @@ "x-looker-nullable": false }, "content_metadata_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of content metadata", "x-looker-nullable": true @@ -29330,8 +28930,7 @@ "x-looker-nullable": true }, "creator_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "User Id of Creator", "x-looker-nullable": true @@ -29625,8 +29224,7 @@ "GroupIdForGroupInclusion": { "properties": { "group_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of group", "x-looker-nullable": true @@ -29669,8 +29267,7 @@ "x-looker-nullable": false }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "x-looker-nullable": false @@ -29731,8 +29328,7 @@ "x-looker-nullable": false }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "x-looker-nullable": false @@ -29802,8 +29398,7 @@ "x-looker-nullable": false }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "x-looker-nullable": false @@ -29829,8 +29424,7 @@ "parent_group_ids": { "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" }, "readOnly": true, "description": "IDs of parents of this group", @@ -29839,8 +29433,7 @@ "role_ids": { "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" }, "readOnly": true, "description": "Role IDs assigned to group", @@ -29852,8 +29445,7 @@ "GroupIdForGroupUserInclusion": { "properties": { "user_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of user", "x-looker-nullable": true @@ -29902,8 +29494,7 @@ "x-looker-nullable": false }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "ID of the hub.", "x-looker-nullable": false @@ -29982,8 +29573,7 @@ "x-looker-nullable": false }, "integration_hub_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "ID of the integration hub.", "x-looker-nullable": false @@ -30046,9 +29636,10 @@ "x-looker-values": [ "cell", "query", - "dashboard" + "dashboard", + "none" ], - "description": "A list of action types the integration supports. Valid values are: \"cell\", \"query\", \"dashboard\".", + "description": "A list of action types the integration supports. Valid values are: \"cell\", \"query\", \"dashboard\", \"none\".", "x-looker-nullable": false }, "supported_formattings": { @@ -30120,8 +29711,7 @@ "installed_delegate_oauth_targets": { "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" }, "description": "Whether the integration is available to users.", "x-looker-nullable": false @@ -30344,8 +29934,7 @@ "default_new_user_group_ids": { "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" }, "x-looker-write-only": true, "description": "(Write-Only) Array of ids of groups that will be applied to new users the first time they login via LDAP", @@ -30363,8 +29952,7 @@ "default_new_user_role_ids": { "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" }, "x-looker-write-only": true, "description": "(Write-Only) Array of ids of roles that will be applied to new users the first time they login via LDAP", @@ -30624,15 +30212,13 @@ "LDAPGroupRead": { "properties": { "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "x-looker-nullable": false }, "looker_group_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id of group in Looker", "x-looker-nullable": true @@ -30671,14 +30257,12 @@ "LDAPGroupWrite": { "properties": { "id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "Unique Id", "x-looker-nullable": true }, "looker_group_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id of group in Looker", "x-looker-nullable": true @@ -30696,8 +30280,7 @@ "role_ids": { "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" }, "description": "Looker Role Ids", "x-looker-nullable": true @@ -30760,8 +30343,7 @@ "user_attribute_ids": { "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" }, "description": "Looker User Attribute Ids", "x-looker-nullable": true @@ -30993,15 +30575,13 @@ "x-looker-nullable": false }, "content_metadata_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of content metadata", "x-looker-nullable": true }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "x-looker-nullable": false @@ -31013,8 +30593,7 @@ "x-looker-nullable": true }, "user_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "User Id", "x-looker-nullable": true } @@ -31033,8 +30612,7 @@ "x-looker-nullable": false }, "content_metadata_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of content metadata", "x-looker-nullable": true @@ -31051,14 +30629,12 @@ "x-looker-nullable": true }, "user_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "User Id", "x-looker-nullable": true }, "content_favorite_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Content Favorite Id", "x-looker-nullable": true @@ -31083,8 +30659,7 @@ "x-looker-nullable": true }, "deleter_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of User that deleted the look.", "x-looker-nullable": true @@ -31138,8 +30713,7 @@ "x-looker-nullable": true }, "last_updater_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of User that last updated the look.", "x-looker-nullable": true @@ -31175,8 +30749,7 @@ "x-looker-nullable": true }, "query_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "Query Id", "x-looker-nullable": true }, @@ -31226,8 +30799,7 @@ "x-looker-nullable": false }, "content_metadata_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of content metadata", "x-looker-nullable": true @@ -31244,14 +30816,12 @@ "x-looker-nullable": true }, "user_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "User Id", "x-looker-nullable": true }, "content_favorite_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Content Favorite Id", "x-looker-nullable": true @@ -31276,8 +30846,7 @@ "x-looker-nullable": true }, "deleter_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of User that deleted the look.", "x-looker-nullable": true @@ -31331,8 +30900,7 @@ "x-looker-nullable": true }, "last_updater_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of User that last updated the look.", "x-looker-nullable": true @@ -31368,8 +30936,7 @@ "x-looker-nullable": true }, "query_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "Query Id", "x-looker-nullable": true }, @@ -31431,8 +30998,7 @@ "x-looker-nullable": false }, "content_metadata_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of content metadata", "x-looker-nullable": true @@ -31449,14 +31015,12 @@ "x-looker-nullable": true }, "user_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "User Id", "x-looker-nullable": true }, "content_favorite_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Content Favorite Id", "x-looker-nullable": true @@ -31481,8 +31045,7 @@ "x-looker-nullable": true }, "deleter_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of User that deleted the look.", "x-looker-nullable": true @@ -31536,8 +31099,7 @@ "x-looker-nullable": true }, "last_updater_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of User that last updated the look.", "x-looker-nullable": true @@ -31573,8 +31135,7 @@ "x-looker-nullable": true }, "query_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "Query Id", "x-looker-nullable": true }, @@ -33034,8 +32595,7 @@ "x-looker-nullable": true }, "result_maker_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique to get results", "x-looker-nullable": true @@ -33088,8 +32648,7 @@ "x-looker-nullable": true }, "query_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "Id of the query to merge", "x-looker-nullable": true } @@ -33150,8 +32709,7 @@ "x-looker-nullable": false }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "x-looker-nullable": false @@ -33216,8 +32774,7 @@ "x-looker-nullable": false }, "group_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "If set, only Looker users who are members of this group can use this web app with Looker. If group_id is not set, any Looker user may use this app to access this Looker instance", "x-looker-nullable": true }, @@ -33275,8 +32832,7 @@ "default_new_user_group_ids": { "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" }, "x-looker-write-only": true, "description": "(Write-Only) Array of ids of groups that will be applied to new users the first time they login via OIDC", @@ -33294,8 +32850,7 @@ "default_new_user_role_ids": { "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" }, "x-looker-write-only": true, "description": "(Write-Only) Array of ids of roles that will be applied to new users the first time they login via OIDC", @@ -33355,8 +32910,7 @@ "x-looker-nullable": true }, "modified_by": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "User id of user who last modified this config", "x-looker-nullable": true @@ -33462,15 +33016,13 @@ "OIDCGroupRead": { "properties": { "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "x-looker-nullable": false }, "looker_group_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id of group in Looker", "x-looker-nullable": true @@ -33502,14 +33054,12 @@ "OIDCGroupWrite": { "properties": { "id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "Unique Id", "x-looker-nullable": true }, "looker_group_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id of group in Looker", "x-looker-nullable": true @@ -33527,8 +33077,7 @@ "role_ids": { "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" }, "description": "Looker Role Ids", "x-looker-nullable": true @@ -33577,8 +33126,7 @@ "user_attribute_ids": { "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" }, "description": "Looker User Attribute Ids", "x-looker-nullable": true @@ -33675,8 +33223,7 @@ "x-looker-nullable": false }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "x-looker-nullable": false @@ -34148,8 +33695,7 @@ "x-looker-nullable": false }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "x-looker-nullable": false @@ -34324,8 +33870,7 @@ "x-looker-nullable": false }, "query_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "Id of query to run", "x-looker-nullable": true }, @@ -34391,8 +33936,7 @@ "x-looker-nullable": false }, "query_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "Id of query", "x-looker-nullable": true }, @@ -34509,8 +34053,7 @@ "x-looker-nullable": true }, "dashboard_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of dashboard to render", "x-looker-nullable": true @@ -34553,8 +34096,7 @@ "x-looker-nullable": true }, "query_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of query to render", "x-looker-nullable": true @@ -34605,8 +34147,7 @@ "x-looker-nullable": true }, "user_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "The user account permissions in which the render task will execute", "x-looker-nullable": true @@ -34740,8 +34281,7 @@ "ResultMakerWithIdVisConfigAndDynamicFields": { "properties": { "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id.", "x-looker-nullable": false @@ -34783,8 +34323,7 @@ "x-looker-nullable": false }, "query_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "ID of query if this is a query.", "x-looker-nullable": true @@ -34826,8 +34365,7 @@ "x-looker-nullable": false }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "x-looker-nullable": false @@ -34844,8 +34382,7 @@ "x-looker-nullable": true }, "permission_set_id": { - "type": "integer", - "format": "int64", + "type": "string", "x-looker-write-only": true, "description": "(Write-Only) Id of permission set", "x-looker-nullable": true @@ -34857,8 +34394,7 @@ "x-looker-nullable": true }, "model_set_id": { - "type": "integer", - "format": "int64", + "type": "string", "x-looker-write-only": true, "description": "(Write-Only) Id of model set", "x-looker-nullable": true @@ -34892,8 +34428,7 @@ "x-looker-nullable": false }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "x-looker-nullable": false @@ -34910,8 +34445,7 @@ "x-looker-nullable": true }, "permission_set_id": { - "type": "integer", - "format": "int64", + "type": "string", "x-looker-write-only": true, "description": "(Write-Only) Id of permission set", "x-looker-nullable": true @@ -34923,8 +34457,7 @@ "x-looker-nullable": true }, "model_set_id": { - "type": "integer", - "format": "int64", + "type": "string", "x-looker-write-only": true, "description": "(Write-Only) Id of model set", "x-looker-nullable": true @@ -34965,8 +34498,7 @@ "x-looker-nullable": false }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "x-looker-nullable": false @@ -35195,8 +34727,7 @@ "default_new_user_role_ids": { "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" }, "x-looker-write-only": true, "description": "(Write-Only) Array of ids of roles that will be applied to new users the first time they login via Saml", @@ -35205,8 +34736,7 @@ "default_new_user_group_ids": { "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" }, "x-looker-write-only": true, "description": "(Write-Only) Array of ids of groups that will be applied to new users the first time they login via Saml", @@ -35304,15 +34834,13 @@ "SamlGroupRead": { "properties": { "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "x-looker-nullable": false }, "looker_group_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id of group in Looker", "x-looker-nullable": true @@ -35351,14 +34879,12 @@ "SamlGroupWrite": { "properties": { "id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "Unique Id", "x-looker-nullable": true }, "looker_group_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id of group in Looker", "x-looker-nullable": true @@ -35376,8 +34902,7 @@ "role_ids": { "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" }, "description": "Looker Role Ids", "x-looker-nullable": true @@ -35472,8 +34997,7 @@ "user_attribute_ids": { "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" }, "description": "Looker User Attribute Ids", "x-looker-nullable": true @@ -35491,15 +35015,13 @@ "ScheduledPlanDestination": { "properties": { "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "x-looker-nullable": false }, "scheduled_plan_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "Id of a scheduled plan you own", "x-looker-nullable": true }, @@ -35561,8 +35083,7 @@ "x-looker-nullable": true }, "user_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "User Id which owns this scheduled plan", "x-looker-nullable": true }, @@ -35582,8 +35103,7 @@ "x-looker-nullable": true }, "dashboard_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "Id of a dashboard", "x-looker-nullable": true }, @@ -35703,8 +35223,7 @@ "x-looker-nullable": true }, "user_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "User Id which owns this scheduled plan", "x-looker-nullable": true }, @@ -35724,8 +35243,7 @@ "x-looker-nullable": true }, "dashboard_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "Id of a dashboard", "x-looker-nullable": true }, @@ -35835,8 +35353,7 @@ "x-looker-nullable": true }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "x-looker-nullable": false @@ -36392,8 +35909,7 @@ "x-looker-nullable": false }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "x-looker-nullable": false @@ -36454,8 +35970,7 @@ "x-looker-nullable": true }, "sudo_user_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Actual user in the case when this session represents one user sudo'ing as another", "x-looker-nullable": true @@ -36499,10 +36014,10 @@ "description": "Toggle marketplace on or off", "x-looker-nullable": false }, - "whitelabel_configuration": { - "$ref": "#/definitions/WhitelabelConfiguration", + "privatelabel_configuration": { + "$ref": "#/definitions/PrivatelabelConfiguration", "readOnly": true, - "description": "Whitelabel configuration", + "description": "Private label configuration", "x-looker-nullable": false }, "custom_welcome_email": { @@ -36717,8 +36232,7 @@ "x-looker-nullable": true }, "result_maker_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "ID of the ResultMakerLookup entry.", "x-looker-nullable": true } @@ -37061,8 +36575,7 @@ "x-looker-nullable": true }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "x-looker-nullable": false @@ -37115,8 +36628,7 @@ "x-looker-nullable": false }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "x-looker-nullable": false @@ -37193,22 +36705,19 @@ "x-looker-nullable": false }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id of this group-attribute relation", "x-looker-nullable": false }, "group_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of group", "x-looker-nullable": true }, "user_attribute_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of user attribute", "x-looker-nullable": true @@ -37271,8 +36780,7 @@ "x-looker-nullable": true }, "user_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of User", "x-looker-nullable": true @@ -37290,8 +36798,7 @@ "x-looker-nullable": false }, "user_attribute_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of User Attribute", "x-looker-nullable": true @@ -37354,8 +36861,7 @@ "x-looker-nullable": true }, "user_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "User ID", "x-looker-nullable": true @@ -37493,10 +36999,10 @@ "x-looker-nullable": true }, "embed_group_space_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, - "description": "(Embed only) ID of user's group space based on the external_group_id optionally specified during embed user login", + "x-looker-deprecated": true, + "description": "(DEPRECATED) (Embed only) ID of user's group space based on the external_group_id optionally specified during embed user login", "x-looker-nullable": true }, "first_name": { @@ -37507,8 +37013,7 @@ "group_ids": { "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" }, "readOnly": true, "description": "Array of ids of the groups for this user", @@ -37520,8 +37025,7 @@ "x-looker-nullable": true }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "x-looker-nullable": false @@ -37556,8 +37060,7 @@ "x-looker-nullable": true }, "personal_folder_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "ID of user's personal folder", "x-looker-nullable": true @@ -37571,8 +37074,7 @@ "role_ids": { "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" }, "readOnly": true, "description": "Array of ids of the roles for this user", @@ -37625,6 +37127,12 @@ "description": "User can inherit roles from a normal Looker group.", "x-looker-nullable": false }, + "embed_group_folder_id": { + "type": "string", + "readOnly": true, + "description": "(Embed only) ID of user's group folder based on the external_group_id optionally specified during embed user login", + "x-looker-nullable": true + }, "url": { "type": "string", "format": "uri-reference", @@ -37647,8 +37155,7 @@ "x-looker-nullable": false }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "x-looker-nullable": false @@ -37779,8 +37286,7 @@ "WhitelabelConfiguration": { "properties": { "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "x-looker-nullable": false @@ -37865,6 +37371,88 @@ }, "x-looker-status": "stable" }, + "PrivatelabelConfiguration": { + "properties": { + "logo_file": { + "type": "string", + "description": "Customer logo image. Expected base64 encoded data (write-only)", + "x-looker-nullable": true + }, + "logo_url": { + "type": "string", + "readOnly": true, + "description": "Logo image url (read-only)", + "x-looker-nullable": true + }, + "favicon_file": { + "type": "string", + "description": "Custom favicon image. Expected base64 encoded data (write-only)", + "x-looker-nullable": true + }, + "favicon_url": { + "type": "string", + "readOnly": true, + "description": "Favicon image url (read-only)", + "x-looker-nullable": true + }, + "default_title": { + "type": "string", + "description": "Default page title", + "x-looker-nullable": true + }, + "show_help_menu": { + "type": "boolean", + "description": "Boolean to toggle showing help menus", + "x-looker-nullable": false + }, + "show_docs": { + "type": "boolean", + "description": "Boolean to toggle showing docs", + "x-looker-nullable": false + }, + "show_email_sub_options": { + "type": "boolean", + "description": "Boolean to toggle showing email subscription options.", + "x-looker-nullable": false + }, + "allow_looker_mentions": { + "type": "boolean", + "description": "Boolean to toggle mentions of Looker in emails", + "x-looker-nullable": false + }, + "allow_looker_links": { + "type": "boolean", + "description": "Boolean to toggle links to Looker in emails", + "x-looker-nullable": false + }, + "custom_welcome_email_advanced": { + "type": "boolean", + "description": "Allow subject line and email heading customization in customized emails”", + "x-looker-nullable": false + }, + "setup_mentions": { + "type": "boolean", + "description": "Remove the word Looker from appearing in the account setup page", + "x-looker-nullable": false + }, + "alerts_logo": { + "type": "boolean", + "description": "Remove Looker logo from Alerts", + "x-looker-nullable": false + }, + "alerts_links": { + "type": "boolean", + "description": "Remove Looker links from Alerts", + "x-looker-nullable": false + }, + "folders_mentions": { + "type": "boolean", + "description": "Remove Looker mentions in home folder page when you don’t have any items saved", + "x-looker-nullable": false + } + }, + "x-looker-status": "stable" + }, "Workspace": { "properties": { "can": { diff --git a/spec/Looker.4.0.oas.json b/spec/Looker.4.0.oas.json index 6d8a02e9c..dd7b11854 100644 --- a/spec/Looker.4.0.oas.json +++ b/spec/Looker.4.0.oas.json @@ -1,17 +1,17 @@ { "openapi": "3.0.0", "info": { - "version": "4.0.22.2", - "x-looker-release-version": "22.2.3", - "title": "Looker API 4.0 (Beta) Reference", - "description": "\nWelcome to the future! API 4.0 co-exists with APIs 3.1 and 3.0. (3.0 should no longer be used.)\nThe \"beta\" tag means updates for API 4.0 may include breaking changes, but as always we will work to minimize them.\n\n### Authorization\n\nThe classic method of API authorization uses Looker **API3** credentials for authorization and access control.\nLooker admins can create API3 credentials on Looker's **Admin/Users** page.\n\nAPI 4.0 adds additional ways to authenticate API requests, including OAuth and CORS requests.\n\nFor details, see [Looker API Authorization](https://looker.com/docs/r/api/authorization).\n\n\n### API Explorer\n\nThe API Explorer is a Looker-provided utility with many new and unique features for learning and using the Looker API and SDKs.\nIt is a replacement for the 'api-docs' page currently provided on Looker instances.\n\nFor details, see the [API Explorer documentation](https://looker.com/docs/r/api/explorer).\n\n\n### Looker Language SDKs\n\nThe Looker API is a RESTful system that should be usable by any programming language capable of making\nHTTPS requests. SDKs for a variety of programming languages are also provided to streamline using the API. Looker\nhas an OpenSource [sdk-codegen project](https://github.com/looker-open-source/sdk-codegen) that provides several\nlanguage SDKs. Language SDKs generated by `sdk-codegen` have an Authentication manager that can automatically\nauthenticate API requests when needed.\n\nFor details on available Looker SDKs, see [Looker API Client SDKs](https://looker.com/docs/r/api/client_sdks).\n\n\n### API Versioning\n\nFuture releases of Looker expand the latest API version release-by-release to securely expose more and more of the core\npower of the Looker platform to API client applications. API endpoints marked as \"beta\" may receive breaking changes without\nwarning (but we will try to avoid doing that). Stable (non-beta) API endpoints should not receive breaking\nchanges in future releases.\n\nFor details, see [Looker API Versioning](https://looker.com/docs/r/api/versioning).\n\n\n### In This Release\n\nAPI 4.0 version was introduced so we can make adjustments to API functions, parameters, and response types to\nfix bugs and inconsistencies. These changes fall outside the bounds of non-breaking additive changes we can\nmake to our stable API 3.1.\n\nOne benefit of these type adjustments in API 4.0 is dramatically better support for strongly\ntyped languages like TypeScript, Kotlin, Swift, Go, C#, and more.\n\nWhile API 3.1 is still the de-facto Looker API (\"current\", \"stable\", \"default\", etc), the bulk\nof our development activity has shifted to API 4.0, where all new features are added.\n\nThe API Explorer can be used to [interactively compare](https://looker.com/docs/r/api/explorer#comparing_api_versions) the differences between API 3.1 and 4.0.\n\n\n### API and SDK Support Policies\n\nLooker API versions and language SDKs have varying support levels. Please read the API and SDK\n[support policies](https://looker.com/docs/r/api/support-policy) for more information.\n\n\n", + "version": "4.0.22.3", + "x-looker-release-version": "22.3.0", + "title": "Looker API 4.0 Reference", + "description": "\nAPI 4.0 is the current release of the Looker API. API 3.1 is deprecated.\n\n### Authorization\n\nThe classic method of API authorization uses Looker **API3** credentials for authorization and access control.\nLooker admins can create API3 credentials on Looker's **Admin/Users** page.\n\nAPI 4.0 adds additional ways to authenticate API requests, including OAuth and CORS requests.\n\nFor details, see [Looker API Authorization](https://looker.com/docs/r/api/authorization).\n\n\n### API Explorer\n\nThe API Explorer is a Looker-provided utility with many new and unique features for learning and using the Looker API and SDKs.\n\nFor details, see the [API Explorer documentation](https://looker.com/docs/r/api/explorer).\n\n\n### Looker Language SDKs\n\nThe Looker API is a RESTful system that should be usable by any programming language capable of making\nHTTPS requests. SDKs for a variety of programming languages are also provided to streamline using the API. Looker\nhas an OpenSource [sdk-codegen project](https://github.com/looker-open-source/sdk-codegen) that provides several\nlanguage SDKs. Language SDKs generated by `sdk-codegen` have an Authentication manager that can automatically\nauthenticate API requests when needed.\n\nFor details on available Looker SDKs, see [Looker API Client SDKs](https://looker.com/docs/r/api/client_sdks).\n\n\n### API Versioning\n\nFuture releases of Looker expand the latest API version release-by-release to securely expose more and more of the core\npower of the Looker platform to API client applications. API endpoints marked as \"beta\" may receive breaking changes without\nwarning (but we will try to avoid doing that). Stable (non-beta) API endpoints should not receive breaking\nchanges in future releases.\n\nFor details, see [Looker API Versioning](https://looker.com/docs/r/api/versioning).\n\n\n### In This Release\n\nAPI 4.0 version was introduced to make adjustments to API functions, parameters, and response types to\nfix bugs and inconsistencies. These changes fall outside the bounds of non-breaking additive changes we can\nmake to the previous API 3.1.\n\nOne benefit of these type adjustments in API 4.0 is dramatically better support for strongly\ntyped languages like TypeScript, Kotlin, Swift, Go, C#, and more.\n\nSee the [API 4.0 GA announcement](https://developers.looker.com/api/advanced-usage/version-4-ga) for more information\nabout API 4.0.\n\nThe API Explorer can be used to [interactively compare](https://looker.com/docs/r/api/explorer#comparing_api_versions) the differences between API 3.1 and 4.0.\n\n\n### API and SDK Support Policies\n\nLooker API versions and language SDKs have varying support levels. Please read the API and SDK\n[support policies](https://looker.com/docs/r/api/support-policy) for more information.\n\n\n", "contact": { "name": "Looker Team", "url": "https://help.looker.com" }, "license": { "name": "EULA", - "url": "https://localhost:10000/eula" + "url": "https://self-signed.looker.com:9999/eula" } }, "tags": [ @@ -35,10 +35,6 @@ "name": "ColorCollection", "description": "Manage Color Collections" }, - { - "name": "Command", - "description": "Manage Commands" - }, { "name": "Config", "description": "Manage General Configuration" @@ -572,8 +568,7 @@ "description": "Id of query", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -797,8 +792,7 @@ "description": "Id of query", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -1341,7 +1335,7 @@ ], "operationId": "run_url_encoded_query", "summary": "Run Url Encoded Query", - "description": "### Run an URL encoded query.\n\nThis requires the caller to encode the specifiers for the query into the URL query part using\nLooker-specific syntax as explained below.\n\nGenerally, you would want to use one of the methods that takes the parameters as json in the POST body\nfor creating and/or running queries. This method exists for cases where one really needs to encode the\nparameters into the URL of a single 'GET' request. This matches the way that the Looker UI formats\n'explore' URLs etc.\n\nThe parameters here are very similar to the json body formatting except that the filter syntax is\ntricky. Unfortunately, this format makes this method not currently callable via the 'Try it out!' button\nin this documentation page. But, this is callable when creating URLs manually or when using the Looker SDK.\n\nHere is an example inline query URL:\n\n```\nhttps://looker.mycompany.com:19999/api/3.0/queries/models/thelook/views/inventory_items/run/json?fields=category.name,inventory_items.days_in_inventory_tier,products.count&f[category.name]=socks&sorts=products.count+desc+0&limit=500&query_timezone=America/Los_Angeles\n```\n\nWhen invoking this endpoint with the Ruby SDK, pass the query parameter parts as a hash. The hash to match the above would look like:\n\n```ruby\nquery_params =\n{\n :fields => \"category.name,inventory_items.days_in_inventory_tier,products.count\",\n :\"f[category.name]\" => \"socks\",\n :sorts => \"products.count desc 0\",\n :limit => \"500\",\n :query_timezone => \"America/Los_Angeles\"\n}\nresponse = ruby_sdk.run_url_encoded_query('thelook','inventory_items','json', query_params)\n\n```\n\nAgain, it is generally easier to use the variant of this method that passes the full query in the POST body.\nThis method is available for cases where other alternatives won't fit the need.\n\nSupported formats:\n\n| result_format | Description\n| :-----------: | :--- |\n| json | Plain json\n| json_detail | Row data plus metadata describing the fields, pivots, table calcs, and other aspects of the query\n| csv | Comma separated values with a header\n| txt | Tab separated values with a header\n| html | Simple html\n| md | Simple markdown\n| xlsx | MS Excel spreadsheet\n| sql | Returns the generated SQL rather than running the query\n| png | A PNG image of the visualization of the query\n| jpg | A JPG image of the visualization of the query\n\n\n", + "description": "### Run an URL encoded query.\n\nThis requires the caller to encode the specifiers for the query into the URL query part using\nLooker-specific syntax as explained below.\n\nGenerally, you would want to use one of the methods that takes the parameters as json in the POST body\nfor creating and/or running queries. This method exists for cases where one really needs to encode the\nparameters into the URL of a single 'GET' request. This matches the way that the Looker UI formats\n'explore' URLs etc.\n\nThe parameters here are very similar to the json body formatting except that the filter syntax is\ntricky. Unfortunately, this format makes this method not currently callable via the 'Try it out!' button\nin this documentation page. But, this is callable when creating URLs manually or when using the Looker SDK.\n\nHere is an example inline query URL:\n\n```\nhttps://looker.mycompany.com:19999/api/3.0/queries/models/thelook/views/inventory_items/run/json?fields=category.name,inventory_items.days_in_inventory_tier,products.count&f[category.name]=socks&sorts=products.count+desc+0&limit=500&query_timezone=America/Los_Angeles\n```\n\nWhen invoking this endpoint with the Ruby SDK, pass the query parameter parts as a hash. The hash to match the above would look like:\n\n```ruby\nquery_params =\n{\n fields: \"category.name,inventory_items.days_in_inventory_tier,products.count\",\n :\"f[category.name]\" => \"socks\",\n sorts: \"products.count desc 0\",\n limit: \"500\",\n query_timezone: \"America/Los_Angeles\"\n}\nresponse = ruby_sdk.run_url_encoded_query('thelook','inventory_items','json', query_params)\n\n```\n\nAgain, it is generally easier to use the variant of this method that passes the full query in the POST body.\nThis method is available for cases where other alternatives won't fit the need.\n\nSupported formats:\n\n| result_format | Description\n| :-----------: | :--- |\n| json | Plain json\n| json_detail | Row data plus metadata describing the fields, pivots, table calcs, and other aspects of the query\n| csv | Comma separated values with a header\n| txt | Tab separated values with a header\n| html | Simple html\n| md | Simple markdown\n| xlsx | MS Excel spreadsheet\n| sql | Returns the generated SQL rather than running the query\n| png | A PNG image of the visualization of the query\n| jpg | A JPG image of the visualization of the query\n\n\n", "parameters": [ { "name": "model_name", @@ -1581,8 +1575,7 @@ "description": "Id of user.", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -1841,8 +1834,7 @@ "description": "ID of an alert", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } } ], @@ -1895,8 +1887,7 @@ "description": "ID of an alert", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } } ], @@ -1990,8 +1981,7 @@ "description": "ID of an alert", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } } ], @@ -2075,8 +2065,7 @@ "description": "ID of an alert", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } } ], @@ -2239,8 +2228,7 @@ "description": "ID of an alert", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -3028,136 +3016,126 @@ "x-looker-activity-type": "non_query" } }, - "/commands": { - "post": { + "/content_favorite/search": { + "get": { "tags": [ - "Command" + "Content" ], - "operationId": "create_command", - "summary": "Create a custom command", - "description": "### Create a new command.\n# Required fields: [:name, :linked_content_id, :linked_content_type]\n# `linked_content_type` must be one of [\"dashboard\", \"lookml_dashboard\"]\n#\n", - "responses": { - "200": { - "description": "The command is saved.", - "content": { - "application/json": { - "schema": { - "$ref": "#/components/schemas/Command" - } - } + "operationId": "search_content_favorites", + "summary": "Search Favorite Contents", + "description": "### Search Favorite Content\n\nIf multiple search params are given and `filter_or` is FALSE or not specified,\nsearch params are combined in a logical AND operation.\nOnly rows that match *all* search param criteria will be returned.\n\nIf `filter_or` is TRUE, multiple search params are combined in a logical OR operation.\nResults will include rows that match **any** of the search criteria.\n\nString search params use case-insensitive matching.\nString search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions.\nexample=\"dan%\" will match \"danger\" and \"Danzig\" but not \"David\"\nexample=\"D_m%\" will match \"Damage\" and \"dump\"\n\nInteger search params can accept a single value or a comma separated list of values. The multiple\nvalues will be combined under a logical OR operation - results will match at least one of\nthe given values.\n\nMost search params can accept \"IS NULL\" and \"NOT NULL\" as special expressions to match\nor exclude (respectively) rows where the column is null.\n\nBoolean search params accept only \"true\" and \"false\" as values.\n\n", + "parameters": [ + { + "name": "id", + "in": "query", + "description": "Match content favorite id(s)", + "required": false, + "schema": { + "type": "string" } }, - "400": { - "description": "Bad Request", - "content": { - "application/json": { - "schema": { - "$ref": "#/components/schemas/Error" - } - } + { + "name": "user_id", + "in": "query", + "description": "Match user id(s).To create a list of multiple ids, use commas as separators", + "required": false, + "schema": { + "type": "string" } }, - "404": { - "description": "Not Found", - "content": { - "application/json": { - "schema": { - "$ref": "#/components/schemas/Error" - } - } + { + "name": "content_metadata_id", + "in": "query", + "description": "Match content metadata id(s).To create a list of multiple ids, use commas as separators", + "required": false, + "schema": { + "type": "string" } }, - "409": { - "description": "Resource Already Exists", - "content": { - "application/json": { - "schema": { - "$ref": "#/components/schemas/Error" - } - } + { + "name": "dashboard_id", + "in": "query", + "description": "Match dashboard id(s).To create a list of multiple ids, use commas as separators", + "required": false, + "schema": { + "type": "string" } }, - "422": { - "description": "Validation Error", - "content": { - "application/json": { - "schema": { - "$ref": "#/components/schemas/ValidationError" - } - } + { + "name": "look_id", + "in": "query", + "description": "Match look id(s).To create a list of multiple ids, use commas as separators", + "required": false, + "schema": { + "type": "string" } }, - "429": { - "description": "Too Many Requests", - "content": { - "application/json": { - "schema": { - "$ref": "#/components/schemas/Error" - } - } + { + "name": "board_id", + "in": "query", + "description": "Match board id(s).To create a list of multiple ids, use commas as separators", + "required": false, + "schema": { + "type": "string" } - } - }, - "x-looker-status": "beta", - "x-looker-activity-type": "non_query", - "requestBody": { - "content": { - "application/json": { - "schema": { - "$ref": "#/components/schemas/Command" - } + }, + { + "name": "limit", + "in": "query", + "description": "Number of results to return. (used with offset)", + "required": false, + "schema": { + "type": "integer", + "format": "int64" + } + }, + { + "name": "offset", + "in": "query", + "description": "Number of results to skip before returning any. (used with limit)", + "required": false, + "schema": { + "type": "integer", + "format": "int64" } }, - "description": "Writable command parameters", - "required": true - } - }, - "get": { - "tags": [ - "Command" - ], - "operationId": "get_all_commands", - "summary": "Get All Commands", - "description": "### Get All Commands.\n", - "parameters": [ { - "name": "content_id", + "name": "sorts", "in": "query", - "description": "Id of the associated content. This must be accompanied with content_type.", + "description": "Fields to sort by.", "required": false, "schema": { "type": "string" } }, { - "name": "content_type", + "name": "fields", "in": "query", - "description": "Type of the associated content. This must be accompanied with content_id.", + "description": "Requested fields.", "required": false, "schema": { "type": "string" } }, { - "name": "limit", + "name": "filter_or", "in": "query", - "description": "Number of results to return.", + "description": "Combine given search criteria in a boolean OR expression", "required": false, "schema": { - "type": "integer", - "format": "int64" + "type": "boolean" } } ], "responses": { "200": { - "description": "Commands", + "description": "Favorite Content", "content": { "application/json": { "schema": { "type": "array", "items": { - "$ref": "#/components/schemas/Command" + "$ref": "#/components/schemas/ContentFavorite" } } } @@ -3182,49 +3160,47 @@ } } } - }, - "405": { - "description": "Resource Can't Be Modified", - "content": { - "application/json": { - "schema": { - "$ref": "#/components/schemas/Error" - } - } - } } }, "x-looker-status": "beta", "x-looker-activity-type": "non_query" } }, - "/commands/{command_id}": { - "patch": { + "/content_favorite/{content_favorite_id}": { + "get": { "tags": [ - "Command" + "Content" ], - "operationId": "update_command", - "summary": "Update a custom command", - "description": "### Update an existing custom command.\n# Optional fields: ['name', 'description']\n#\n", + "operationId": "content_favorite", + "summary": "Get Favorite Content", + "description": "### Get favorite content by its id", "parameters": [ { - "name": "command_id", + "name": "content_favorite_id", "in": "path", - "description": "ID of a command", + "description": "Id of favorite content", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" + } + }, + { + "name": "fields", + "in": "query", + "description": "Requested fields.", + "required": false, + "schema": { + "type": "string" } } ], "responses": { "200": { - "description": "The command is updated.", + "description": "Favorite Content", "content": { "application/json": { "schema": { - "$ref": "#/components/schemas/Command" + "$ref": "#/components/schemas/ContentFavorite" } } } @@ -3248,77 +3224,42 @@ } } } - }, - "422": { - "description": "Validation Error", - "content": { - "application/json": { - "schema": { - "$ref": "#/components/schemas/ValidationError" - } - } - } - }, - "429": { - "description": "Too Many Requests", - "content": { - "application/json": { - "schema": { - "$ref": "#/components/schemas/Error" - } - } - } } }, "x-looker-status": "beta", - "x-looker-activity-type": "non_query", - "requestBody": { - "content": { - "application/json": { - "schema": { - "$ref": "#/components/schemas/UpdateCommand" - } - } - }, - "description": "Re-writable command parameters", - "required": true - } + "x-looker-activity-type": "non_query" }, "delete": { "tags": [ - "Command" + "Content" ], - "operationId": "delete_command", - "summary": "Delete a custom command", - "description": "### Delete an existing custom command.\n", + "operationId": "delete_content_favorite", + "summary": "Delete Favorite Content", + "description": "### Delete favorite content", "parameters": [ { - "name": "command_id", + "name": "content_favorite_id", "in": "path", - "description": "ID of a command", + "description": "Id of favorite content", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } } ], "responses": { "204": { - "description": "The command is deleted." - }, - "400": { - "description": "Bad Request", + "description": "Successfully deleted.", "content": { "application/json": { "schema": { - "$ref": "#/components/schemas/Error" + "type": "string" } } } }, - "404": { - "description": "Not Found", + "400": { + "description": "Bad Request", "content": { "application/json": { "schema": { @@ -3327,288 +3268,8 @@ } } }, - "405": { - "description": "Resource Can't Be Modified", - "content": { - "application/json": { - "schema": { - "$ref": "#/components/schemas/Error" - } - } - } - }, - "429": { - "description": "Too Many Requests", - "content": { - "application/json": { - "schema": { - "$ref": "#/components/schemas/Error" - } - } - } - } - }, - "x-looker-status": "beta", - "x-looker-activity-type": "non_query" - } - }, - "/content_favorite/search": { - "get": { - "tags": [ - "Content" - ], - "operationId": "search_content_favorites", - "summary": "Search Favorite Contents", - "description": "### Search Favorite Content\n\nIf multiple search params are given and `filter_or` is FALSE or not specified,\nsearch params are combined in a logical AND operation.\nOnly rows that match *all* search param criteria will be returned.\n\nIf `filter_or` is TRUE, multiple search params are combined in a logical OR operation.\nResults will include rows that match **any** of the search criteria.\n\nString search params use case-insensitive matching.\nString search params can contain `%` and '_' as SQL LIKE pattern match wildcard expressions.\nexample=\"dan%\" will match \"danger\" and \"Danzig\" but not \"David\"\nexample=\"D_m%\" will match \"Damage\" and \"dump\"\n\nInteger search params can accept a single value or a comma separated list of values. The multiple\nvalues will be combined under a logical OR operation - results will match at least one of\nthe given values.\n\nMost search params can accept \"IS NULL\" and \"NOT NULL\" as special expressions to match\nor exclude (respectively) rows where the column is null.\n\nBoolean search params accept only \"true\" and \"false\" as values.\n\n", - "parameters": [ - { - "name": "id", - "in": "query", - "description": "Match content favorite id(s)", - "required": false, - "schema": { - "type": "integer", - "format": "int64" - } - }, - { - "name": "user_id", - "in": "query", - "description": "Match user id(s).To create a list of multiple ids, use commas as separators", - "required": false, - "schema": { - "type": "string" - } - }, - { - "name": "content_metadata_id", - "in": "query", - "description": "Match content metadata id(s).To create a list of multiple ids, use commas as separators", - "required": false, - "schema": { - "type": "string" - } - }, - { - "name": "dashboard_id", - "in": "query", - "description": "Match dashboard id(s).To create a list of multiple ids, use commas as separators", - "required": false, - "schema": { - "type": "string" - } - }, - { - "name": "look_id", - "in": "query", - "description": "Match look id(s).To create a list of multiple ids, use commas as separators", - "required": false, - "schema": { - "type": "string" - } - }, - { - "name": "board_id", - "in": "query", - "description": "Match board id(s).To create a list of multiple ids, use commas as separators", - "required": false, - "schema": { - "type": "string" - } - }, - { - "name": "limit", - "in": "query", - "description": "Number of results to return. (used with offset)", - "required": false, - "schema": { - "type": "integer", - "format": "int64" - } - }, - { - "name": "offset", - "in": "query", - "description": "Number of results to skip before returning any. (used with limit)", - "required": false, - "schema": { - "type": "integer", - "format": "int64" - } - }, - { - "name": "sorts", - "in": "query", - "description": "Fields to sort by.", - "required": false, - "schema": { - "type": "string" - } - }, - { - "name": "fields", - "in": "query", - "description": "Requested fields.", - "required": false, - "schema": { - "type": "string" - } - }, - { - "name": "filter_or", - "in": "query", - "description": "Combine given search criteria in a boolean OR expression", - "required": false, - "schema": { - "type": "boolean" - } - } - ], - "responses": { - "200": { - "description": "Favorite Content", - "content": { - "application/json": { - "schema": { - "type": "array", - "items": { - "$ref": "#/components/schemas/ContentFavorite" - } - } - } - } - }, - "400": { - "description": "Bad Request", - "content": { - "application/json": { - "schema": { - "$ref": "#/components/schemas/Error" - } - } - } - }, - "404": { - "description": "Not Found", - "content": { - "application/json": { - "schema": { - "$ref": "#/components/schemas/Error" - } - } - } - } - }, - "x-looker-status": "beta", - "x-looker-activity-type": "non_query" - } - }, - "/content_favorite/{content_favorite_id}": { - "get": { - "tags": [ - "Content" - ], - "operationId": "content_favorite", - "summary": "Get Favorite Content", - "description": "### Get favorite content by its id", - "parameters": [ - { - "name": "content_favorite_id", - "in": "path", - "description": "Id of favorite content", - "required": true, - "schema": { - "type": "integer", - "format": "int64" - } - }, - { - "name": "fields", - "in": "query", - "description": "Requested fields.", - "required": false, - "schema": { - "type": "string" - } - } - ], - "responses": { - "200": { - "description": "Favorite Content", - "content": { - "application/json": { - "schema": { - "$ref": "#/components/schemas/ContentFavorite" - } - } - } - }, - "400": { - "description": "Bad Request", - "content": { - "application/json": { - "schema": { - "$ref": "#/components/schemas/Error" - } - } - } - }, - "404": { - "description": "Not Found", - "content": { - "application/json": { - "schema": { - "$ref": "#/components/schemas/Error" - } - } - } - } - }, - "x-looker-status": "beta", - "x-looker-activity-type": "non_query" - }, - "delete": { - "tags": [ - "Content" - ], - "operationId": "delete_content_favorite", - "summary": "Delete Favorite Content", - "description": "### Delete favorite content", - "parameters": [ - { - "name": "content_favorite_id", - "in": "path", - "description": "Id of favorite content", - "required": true, - "schema": { - "type": "integer", - "format": "int64" - } - } - ], - "responses": { - "204": { - "description": "Successfully deleted.", - "content": { - "application/json": { - "schema": { - "type": "string" - } - } - } - }, - "400": { - "description": "Bad Request", - "content": { - "application/json": { - "schema": { - "$ref": "#/components/schemas/Error" - } - } - } - }, - "404": { - "description": "Not Found", + "404": { + "description": "Not Found", "content": { "application/json": { "schema": { @@ -3732,8 +3393,7 @@ "description": "Parent space of content.", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -3800,8 +3460,7 @@ "description": "Id of content metadata", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } } ], @@ -3885,8 +3544,7 @@ "description": "Id of content metadata", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -4045,8 +3703,7 @@ "description": "Id of content metadata", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -4197,8 +3854,7 @@ "description": "Id of content metadata access", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } } ], @@ -4661,8 +4317,7 @@ "description": "Match credentials_email id.", "required": false, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -5000,7 +4655,7 @@ ], "operationId": "create_dashboard", "summary": "Create Dashboard", - "description": "### Create a new dashboard\n\nCreates a new dashboard object and returns the details of the newly created dashboard.\n\n`Title`, `user_id`, and `space_id` are all required fields.\n`Space_id` and `user_id` must contain the id of an existing space or user, respectively.\nA dashboard's `title` must be unique within the space in which it resides.\n\nIf you receive a 422 error response when creating a dashboard, be sure to look at the\nresponse body for information about exactly which fields are missing or contain invalid data.\n\nYou can **update** an existing dashboard with [update_dashboard()](#!/Dashboard/update_dashboard)\n\nYou can **permanently delete** an existing dashboard with [delete_dashboard()](#!/Dashboard/delete_dashboard)\n", + "description": "### Create a new dashboard\n\nCreates a new dashboard object and returns the details of the newly created dashboard.\n\n`Title` and `space_id` are required fields.\n`Space_id` must contain the id of an existing space.\nA dashboard's `title` must be unique within the space in which it resides.\n\nIf you receive a 422 error response when creating a dashboard, be sure to look at the\nresponse body for information about exactly which fields are missing or contain invalid data.\n\nYou can **update** an existing dashboard with [update_dashboard()](#!/Dashboard/update_dashboard)\n\nYou can **permanently delete** an existing dashboard with [delete_dashboard()](#!/Dashboard/delete_dashboard)\n", "responses": { "200": { "description": "Dashboard", @@ -5207,8 +4862,9 @@ { "name": "page", "in": "query", - "description": "Requested page.", + "description": "DEPRECATED. Use limit and offset instead. Return only page N of paginated results", "required": false, + "x-looker-deprecated": true, "schema": { "type": "integer", "format": "int64" @@ -5217,8 +4873,9 @@ { "name": "per_page", "in": "query", - "description": "Results per page.", + "description": "DEPRECATED. Use limit and offset instead. Return N rows of data per page", "required": false, + "x-looker-deprecated": true, "schema": { "type": "integer", "format": "int64" @@ -6174,8 +5831,7 @@ "description": "Select elements that refer to a given dashboard id", "required": false, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -6184,8 +5840,7 @@ "description": "Select elements that refer to a given look id", "required": false, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -6577,6 +6232,15 @@ "schema": { "type": "string" } + }, + { + "name": "apply_filters", + "in": "query", + "description": "Apply relevant filters on dashboard to this tile", + "required": false, + "schema": { + "type": "boolean" + } } ], "responses": { @@ -7841,8 +7505,7 @@ "description": "ID of datagroup.", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } } ], @@ -7895,8 +7558,7 @@ "description": "ID of datagroup.", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } } ], @@ -9417,8 +9079,7 @@ "description": "Id of Embed Secret", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } } ], @@ -10321,8 +9982,9 @@ { "name": "page", "in": "query", - "description": "Requested page.", + "description": "DEPRECATED. Use limit and offset instead. Return only page N of paginated results", "required": false, + "x-looker-deprecated": true, "schema": { "type": "integer", "format": "int64" @@ -10331,7 +9993,28 @@ { "name": "per_page", "in": "query", - "description": "Results per page.", + "description": "DEPRECATED. Use limit and offset instead. Return N rows of data per page", + "required": false, + "x-looker-deprecated": true, + "schema": { + "type": "integer", + "format": "int64" + } + }, + { + "name": "limit", + "in": "query", + "description": "Number of results to return. (used with offset and takes priority over page and per_page)", + "required": false, + "schema": { + "type": "integer", + "format": "int64" + } + }, + { + "name": "offset", + "in": "query", + "description": "Number of results to skip before returning any. (used with limit and takes priority over page and per_page)", "required": false, "schema": { "type": "integer", @@ -10357,8 +10040,7 @@ "schema": { "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" } } }, @@ -10368,8 +10050,7 @@ "description": "Id of content metadata to which groups must have access.", "required": false, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -10577,8 +10258,7 @@ "description": "Match group id.", "required": false, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -10719,8 +10399,7 @@ "description": "Match group id.", "required": false, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -10861,8 +10540,7 @@ "description": "Match group id.", "required": false, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -10956,8 +10634,7 @@ "description": "Id of group", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -11019,8 +10696,7 @@ "description": "Id of group", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -11113,8 +10789,7 @@ "description": "Id of group", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } } ], @@ -11189,8 +10864,7 @@ "description": "Id of group", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -11255,8 +10929,7 @@ "description": "Id of group", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } } ], @@ -11332,8 +11005,7 @@ "description": "Id of group", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -11348,8 +11020,9 @@ { "name": "page", "in": "query", - "description": "Requested page.", + "description": "DEPRECATED. Use limit and offset instead. Return only page N of paginated results", "required": false, + "x-looker-deprecated": true, "schema": { "type": "integer", "format": "int64" @@ -11358,7 +11031,28 @@ { "name": "per_page", "in": "query", - "description": "Results per page.", + "description": "DEPRECATED. Use limit and offset instead. Return N rows of data per page", + "required": false, + "x-looker-deprecated": true, + "schema": { + "type": "integer", + "format": "int64" + } + }, + { + "name": "limit", + "in": "query", + "description": "Number of results to return. (used with offset and takes priority over page and per_page)", + "required": false, + "schema": { + "type": "integer", + "format": "int64" + } + }, + { + "name": "offset", + "in": "query", + "description": "Number of results to skip before returning any. (used with limit and takes priority over page and per_page)", "required": false, "schema": { "type": "integer", @@ -11427,8 +11121,7 @@ "description": "Id of group", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } } ], @@ -11504,8 +11197,7 @@ "description": "Id of group", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -11514,8 +11206,7 @@ "description": "Id of user to remove from group", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } } ], @@ -11573,8 +11264,7 @@ "description": "Id of group", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -11583,8 +11273,7 @@ "description": "Id of group to delete", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } } ], @@ -11642,8 +11331,7 @@ "description": "Id of group", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -11652,8 +11340,7 @@ "description": "Id of user attribute", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } } ], @@ -11727,8 +11414,7 @@ "description": "Id of group", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -11737,8 +11423,7 @@ "description": "Id of user attribute", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } } ], @@ -12007,7 +11692,7 @@ { "name": "page", "in": "query", - "description": "The page to return.", + "description": "The page to return. DEPRECATED. Use offset instead.", "required": false, "schema": { "type": "integer", @@ -12017,7 +11702,7 @@ { "name": "per_page", "in": "query", - "description": "The number of items in the returned page.", + "description": "The number of items in the returned page. DEPRECATED. Use limit instead.", "required": false, "schema": { "type": "integer", @@ -12108,8 +11793,7 @@ "description": "Id of board", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -12171,8 +11855,7 @@ "description": "Id of board", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -12265,8 +11948,7 @@ "description": "Id of board", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } } ], @@ -12501,8 +12183,7 @@ "description": "Id of board item", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -12564,8 +12245,7 @@ "description": "Id of board item", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -12655,11 +12335,10 @@ { "name": "board_item_id", "in": "path", - "description": "Id of board_item", + "description": "Id of board item", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } } ], @@ -12943,8 +12622,7 @@ "description": "Id of board section", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -13006,8 +12684,7 @@ "description": "Id of board section", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -13100,8 +12777,7 @@ "description": "Id of board section", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } } ], @@ -13316,11 +12992,10 @@ { "name": "integration_hub_id", "in": "path", - "description": "Id of Integration Hub", + "description": "Id of integration_hub", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -13379,11 +13054,10 @@ { "name": "integration_hub_id", "in": "path", - "description": "Id of Integration Hub", + "description": "Id of integration_hub", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -13477,8 +13151,7 @@ "description": "Id of integration_hub", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } } ], @@ -13543,8 +13216,7 @@ "description": "Id of integration_hub", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } } ], @@ -15027,8 +14699,7 @@ "description": "Select looks that reference a particular query by query_id", "required": false, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -15061,8 +14732,9 @@ { "name": "page", "in": "query", - "description": "Requested page.", + "description": "DEPRECATED. Use limit and offset instead. Return only page N of paginated results", "required": false, + "x-looker-deprecated": true, "schema": { "type": "integer", "format": "int64" @@ -15071,8 +14743,9 @@ { "name": "per_page", "in": "query", - "description": "Results per page.", + "description": "DEPRECATED. Use limit and offset instead. Return N rows of data per page", "required": false, + "x-looker-deprecated": true, "schema": { "type": "integer", "format": "int64" @@ -17412,8 +17085,7 @@ "description": "Match model set id.", "required": false, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -17507,8 +17179,7 @@ "description": "Id of model set", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -17570,14 +17241,13 @@ "description": "id of model set", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } } ], "responses": { "204": { - "description": "Model set succssfully deleted.", + "description": "Model set successfully deleted.", "content": { "application/json": { "schema": { @@ -17634,8 +17304,7 @@ "description": "id of model set", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } } ], @@ -18297,8 +17966,7 @@ "description": "The id of the user to enable use of this app", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -18392,8 +18060,7 @@ "description": "The id of the user to enable use of this app", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -18993,8 +18660,7 @@ "description": "Match permission set id.", "required": false, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -19088,8 +18754,7 @@ "description": "Id of permission set", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -19151,8 +18816,7 @@ "description": "Id of permission set", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } } ], @@ -19222,11 +18886,10 @@ { "name": "permission_set_id", "in": "path", - "description": "id of permission set", + "description": "Id of permission set", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } } ], @@ -21124,8 +20787,7 @@ "description": "Id of look to render", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -21248,8 +20910,7 @@ "description": "Id of the query to render", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -22056,8 +21717,7 @@ "schema": { "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" } } } @@ -22237,8 +21897,7 @@ "description": "Match role id.", "required": false, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -22361,8 +22020,7 @@ "description": "Match role id.", "required": false, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -22447,8 +22105,7 @@ "description": "id of role", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } } ], @@ -22501,8 +22158,7 @@ "description": "id of role", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } } ], @@ -22575,8 +22231,7 @@ "description": "id of role", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } } ], @@ -22672,8 +22327,7 @@ "description": "id of role", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -22735,11 +22389,10 @@ { "name": "role_id", "in": "path", - "description": "Id of Role", + "description": "id of role", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } } ], @@ -22806,8 +22459,7 @@ "schema": { "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" } } } @@ -22829,11 +22481,10 @@ { "name": "role_id", "in": "path", - "description": "id of user", + "description": "id of role", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -22907,8 +22558,7 @@ "description": "id of role", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } } ], @@ -22995,8 +22645,7 @@ "schema": { "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" } } } @@ -23493,8 +23142,7 @@ "description": "Space Id", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -23561,8 +23209,7 @@ "description": "Scheduled Plan Id", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } } ], @@ -23625,8 +23272,7 @@ "description": "Scheduled Plan Id", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } } ], @@ -23710,8 +23356,7 @@ "description": "Scheduled Plan Id", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -23858,8 +23503,7 @@ "description": "Return scheduled plans belonging to this user_id. If not provided, returns scheduled plans owned by the caller.", "required": false, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -24031,8 +23675,7 @@ "description": "Look Id", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -24041,8 +23684,7 @@ "description": "User Id (default is requesting user if not specified)", "required": false, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -24118,8 +23760,7 @@ "description": "Dashboard Id", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -24128,8 +23769,7 @@ "description": "User Id (default is requesting user if not specified)", "required": false, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -24214,8 +23854,7 @@ "description": "User Id (default is requesting user if not specified)", "required": false, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -24291,8 +23930,7 @@ "description": "Id of schedule plan to copy and run", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } } ], @@ -24615,7 +24253,7 @@ ], "operationId": "set_setting", "summary": "Set Setting", - "description": "### Configure Looker Settings\n\nAvailable settings are:\n - extension_framework_enabled\n - marketplace_auto_install_enabled\n - marketplace_enabled\n - whitelabel_configuration\n - custom_welcome_email\n - onboarding_enabled\n\nSee the `Setting` type for more information on the specific values that can be configured.\n", + "description": "### Configure Looker Settings\n\nAvailable settings are:\n - extension_framework_enabled\n - marketplace_auto_install_enabled\n - marketplace_enabled\n - privatelabel_configuration\n - custom_welcome_email\n - onboarding_enabled\n\nSee the `Setting` type for more information on the specific values that can be configured.\n", "parameters": [ { "name": "fields", @@ -24709,7 +24347,7 @@ ], "operationId": "get_setting", "summary": "Get Setting", - "description": "### Get Looker Settings\n\nAvailable settings are:\n - extension_framework_enabled\n - marketplace_auto_install_enabled\n - marketplace_enabled\n - whitelabel_configuration\n - custom_welcome_email\n - onboarding_enabled\n\n", + "description": "### Get Looker Settings\n\nAvailable settings are:\n - extension_framework_enabled\n - marketplace_auto_install_enabled\n - marketplace_enabled\n - privatelabel_configuration\n - custom_welcome_email\n - onboarding_enabled\n\n", "parameters": [ { "name": "fields", @@ -25003,8 +24641,7 @@ "description": "Match Space id", "required": false, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -26727,8 +26364,7 @@ "description": "Match theme id.", "required": false, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -27228,8 +26864,7 @@ "description": "Id of theme", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -27291,8 +26926,7 @@ "description": "Id of theme", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } } ], @@ -28489,8 +28123,7 @@ "description": "Id of user attribute", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -28552,8 +28185,7 @@ "description": "Id of user attribute", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -28643,11 +28275,10 @@ { "name": "user_attribute_id", "in": "path", - "description": "Id of user_attribute", + "description": "Id of user attribute", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } } ], @@ -28712,8 +28343,7 @@ "description": "Id of user attribute", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -28778,8 +28408,7 @@ "description": "Id of user attribute", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } } ], @@ -28945,8 +28574,9 @@ { "name": "page", "in": "query", - "description": "Return only page N of paginated results", + "description": "DEPRECATED. Use limit and offset instead. Return only page N of paginated results", "required": false, + "x-looker-deprecated": true, "schema": { "type": "integer", "format": "int64" @@ -28955,7 +28585,28 @@ { "name": "per_page", "in": "query", - "description": "Return N rows of data per page", + "description": "DEPRECATED. Use limit and offset instead. Return N rows of data per page", + "required": false, + "x-looker-deprecated": true, + "schema": { + "type": "integer", + "format": "int64" + } + }, + { + "name": "limit", + "in": "query", + "description": "Number of results to return. (used with offset and takes priority over page and per_page)", + "required": false, + "schema": { + "type": "integer", + "format": "int64" + } + }, + { + "name": "offset", + "in": "query", + "description": "Number of results to skip before returning any. (used with limit and takes priority over page and per_page)", "required": false, "schema": { "type": "integer", @@ -29189,6 +28840,7 @@ "in": "query", "description": "DEPRECATED. Use limit and offset instead. Return only page N of paginated results", "required": false, + "x-looker-deprecated": true, "schema": { "type": "integer", "format": "int64" @@ -29199,6 +28851,7 @@ "in": "query", "description": "DEPRECATED. Use limit and offset instead. Return N rows of data per page", "required": false, + "x-looker-deprecated": true, "schema": { "type": "integer", "format": "int64" @@ -29243,8 +28896,7 @@ "schema": { "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" } } } @@ -29395,6 +29047,7 @@ "in": "query", "description": "DEPRECATED. Use limit and offset instead. Return only page N of paginated results", "required": false, + "x-looker-deprecated": true, "schema": { "type": "integer", "format": "int64" @@ -29405,6 +29058,7 @@ "in": "query", "description": "DEPRECATED. Use limit and offset instead. Return N rows of data per page", "required": false, + "x-looker-deprecated": true, "schema": { "type": "integer", "format": "int64" @@ -29601,6 +29255,7 @@ "in": "query", "description": "DEPRECATED. Use limit and offset instead. Return only page N of paginated results", "required": false, + "x-looker-deprecated": true, "schema": { "type": "integer", "format": "int64" @@ -29611,6 +29266,7 @@ "in": "query", "description": "DEPRECATED. Use limit and offset instead. Return N rows of data per page", "required": false, + "x-looker-deprecated": true, "schema": { "type": "integer", "format": "int64" @@ -29651,8 +29307,7 @@ "description": "Match User Id", "required": false, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -29755,8 +29410,7 @@ "description": "Id of user", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -29818,8 +29472,7 @@ "description": "Id of user", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -29902,8 +29555,7 @@ "description": "Id of user", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } } ], @@ -30038,11 +29690,10 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -30101,11 +29752,10 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -30205,11 +29855,10 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -30299,11 +29948,10 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } } ], @@ -30365,11 +30013,10 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -30428,11 +30075,10 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -30532,143 +30178,140 @@ { "name": "user_id", "in": "path", - "description": "id of user", - "required": true, - "schema": { - "type": "integer", - "format": "int64" - } - } - ], - "responses": { - "204": { - "description": "Successfully deleted.", - "content": { - "application/json": { - "schema": { - "type": "string" - } - } - } - }, - "400": { - "description": "Bad Request", - "content": { - "application/json": { - "schema": { - "$ref": "#/components/schemas/Error" - } - } - } - }, - "404": { - "description": "Not Found", - "content": { - "application/json": { - "schema": { - "$ref": "#/components/schemas/Error" - } - } - } - }, - "429": { - "description": "Too Many Requests", - "content": { - "application/json": { - "schema": { - "$ref": "#/components/schemas/Error" - } - } - } - } - }, - "x-looker-status": "stable", - "x-looker-activity-type": "non_query" - } - }, - "/users/{user_id}/credentials_ldap": { - "get": { - "tags": [ - "User" - ], - "operationId": "user_credentials_ldap", - "summary": "Get LDAP Credential", - "description": "### LDAP login information for the specified user.", - "parameters": [ - { - "name": "user_id", - "in": "path", - "description": "id of user", + "description": "Id of user", + "required": true, + "schema": { + "type": "string" + } + } + ], + "responses": { + "204": { + "description": "Successfully deleted.", + "content": { + "application/json": { + "schema": { + "type": "string" + } + } + } + }, + "400": { + "description": "Bad Request", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/Error" + } + } + } + }, + "404": { + "description": "Not Found", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/Error" + } + } + } + }, + "429": { + "description": "Too Many Requests", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/Error" + } + } + } + } + }, + "x-looker-status": "stable", + "x-looker-activity-type": "non_query" + } + }, + "/users/{user_id}/credentials_ldap": { + "get": { + "tags": [ + "User" + ], + "operationId": "user_credentials_ldap", + "summary": "Get LDAP Credential", + "description": "### LDAP login information for the specified user.", + "parameters": [ + { + "name": "user_id", + "in": "path", + "description": "Id of user", + "required": true, + "schema": { + "type": "string" + } + }, + { + "name": "fields", + "in": "query", + "description": "Requested fields.", + "required": false, + "schema": { + "type": "string" + } + } + ], + "responses": { + "200": { + "description": "LDAP Credential", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/CredentialsLDAP" + } + } + } + }, + "400": { + "description": "Bad Request", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/Error" + } + } + } + }, + "404": { + "description": "Not Found", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/Error" + } + } + } + } + }, + "x-looker-status": "stable", + "x-looker-activity-type": "non_query" + }, + "delete": { + "tags": [ + "User" + ], + "operationId": "delete_user_credentials_ldap", + "summary": "Delete LDAP Credential", + "description": "### LDAP login information for the specified user.", + "parameters": [ + { + "name": "user_id", + "in": "path", + "description": "Id of user", "required": true, - "schema": { - "type": "integer", - "format": "int64" - } - }, - { - "name": "fields", - "in": "query", - "description": "Requested fields.", - "required": false, "schema": { "type": "string" } } ], - "responses": { - "200": { - "description": "LDAP Credential", - "content": { - "application/json": { - "schema": { - "$ref": "#/components/schemas/CredentialsLDAP" - } - } - } - }, - "400": { - "description": "Bad Request", - "content": { - "application/json": { - "schema": { - "$ref": "#/components/schemas/Error" - } - } - } - }, - "404": { - "description": "Not Found", - "content": { - "application/json": { - "schema": { - "$ref": "#/components/schemas/Error" - } - } - } - } - }, - "x-looker-status": "stable", - "x-looker-activity-type": "non_query" - }, - "delete": { - "tags": [ - "User" - ], - "operationId": "delete_user_credentials_ldap", - "summary": "Delete LDAP Credential", - "description": "### LDAP login information for the specified user.", - "parameters": [ - { - "name": "user_id", - "in": "path", - "description": "id of user", - "required": true, - "schema": { - "type": "integer", - "format": "int64" - } - } - ], "responses": { "204": { "description": "Successfully deleted.", @@ -30727,11 +30370,10 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -30790,11 +30432,10 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } } ], @@ -30856,11 +30497,10 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -30919,11 +30559,10 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } } ], @@ -30985,11 +30624,10 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -31048,11 +30686,10 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } } ], @@ -31117,8 +30754,7 @@ "description": "Id of user", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -31127,8 +30763,7 @@ "description": "Id of API 3 Credential", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -31187,21 +30822,19 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { "name": "credentials_api3_id", "in": "path", - "description": "id of API 3 Credential", + "description": "Id of API 3 Credential", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } } ], @@ -31263,11 +30896,10 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -31329,11 +30961,10 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -31427,8 +31058,7 @@ "description": "Id of user", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -31437,8 +31067,7 @@ "description": "Id of Embedding Credential", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -31497,21 +31126,19 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { "name": "credentials_embed_id", "in": "path", - "description": "id of Embedding Credential", + "description": "Id of Embedding Credential", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } } ], @@ -31573,11 +31200,10 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -31641,11 +31267,10 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -31704,11 +31329,10 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } } ], @@ -31773,8 +31397,7 @@ "description": "Id of user", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -31783,8 +31406,7 @@ "description": "Id of Web Login Session", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -31843,21 +31465,19 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { "name": "session_id", "in": "path", - "description": "id of Web Login Session", + "description": "Id of Web Login Session", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } } ], @@ -31919,11 +31539,10 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -31990,8 +31609,7 @@ "description": "Id of user", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -32061,11 +31679,10 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -32136,11 +31753,10 @@ { "name": "user_id", "in": "path", - "description": "id of user", + "description": "Id of user", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -32196,8 +31812,7 @@ "schema": { "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" } } } @@ -32222,8 +31837,7 @@ "description": "Id of user", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -32245,8 +31859,7 @@ "schema": { "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" } } }, @@ -32313,8 +31926,7 @@ "description": "Id of user", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -32323,8 +31935,7 @@ "description": "Id of user attribute", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } } ], @@ -32398,8 +32009,7 @@ "description": "Id of user", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -32408,8 +32018,7 @@ "description": "Id of user attribute", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } } ], @@ -32457,8 +32066,7 @@ "description": "Id of user", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -32522,8 +32130,7 @@ "description": "Id of user", "required": true, "schema": { - "type": "integer", - "format": "int64" + "type": "string" } }, { @@ -33097,7 +32704,7 @@ }, "servers": [ { - "url": "https://localhost:20000/api/4.0" + "url": "https://self-signed.looker.com:19999/api/4.0" } ], "components": { @@ -33137,15 +32744,13 @@ "nullable": false }, "content_favorite_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Content Favorite Id", "nullable": true }, "content_metadata_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of content metadata", "nullable": true @@ -33206,8 +32811,7 @@ "nullable": true }, "user_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of User", "nullable": true @@ -33258,8 +32862,7 @@ "nullable": true }, "homepage_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "Id reference to parent homepage", "nullable": true }, @@ -33273,8 +32876,7 @@ "nullable": true }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "nullable": false @@ -33288,8 +32890,7 @@ "item_order": { "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" }, "description": "ids of the homepage items in the order they should be displayed", "nullable": true @@ -33314,8 +32915,7 @@ "visible_item_order": { "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" }, "readOnly": true, "description": "ids of the homepage items the user can see in the order they should be displayed", @@ -33342,15 +32942,13 @@ "nullable": true }, "content_favorite_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Content favorite id associated with the item this content is based on", "nullable": true }, "content_metadata_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Content metadata id associated with the item this content is based on", "nullable": true @@ -33389,8 +32987,7 @@ "nullable": true }, "dashboard_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "Dashboard to base this item on", "nullable": true }, @@ -33408,14 +33005,12 @@ "nullable": true }, "homepage_section_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "Associated Homepage Section", "nullable": true }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "nullable": false @@ -33727,8 +33322,7 @@ "nullable": true }, "dashboard_element_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "ID of the dashboard element associated with the alert. Refer to [dashboard_element()](#!/Dashboard/DashboardElement)", "nullable": true }, @@ -33761,8 +33355,7 @@ "nullable": false }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "ID of the alert", "nullable": false @@ -33812,8 +33405,7 @@ "nullable": true }, "owner_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "User id of alert owner", "nullable": false }, @@ -33878,8 +33470,7 @@ "AlertPatch": { "properties": { "owner_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "New owner ID of the alert", "nullable": true }, @@ -33924,8 +33515,7 @@ "nullable": true }, "sudo_user_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "The id of the actual user in the case when this session represents one user sudo'ing as another", "nullable": true @@ -33999,15 +33589,13 @@ "nullable": true }, "content_favorite_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Content favorite id associated with the item this content is based on", "nullable": true }, "content_metadata_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Content metadata id associated with the item this content is based on", "nullable": true @@ -34034,8 +33622,7 @@ "nullable": true }, "dashboard_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "Dashboard to base this item on", "nullable": true }, @@ -34053,14 +33640,12 @@ "nullable": true }, "board_section_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "Associated Board Section", "nullable": true }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "nullable": false @@ -34105,12 +33690,44 @@ "description": "Relative url for the associated content", "nullable": false }, + "use_custom_description": { + "type": "boolean", + "description": "Whether the custom description should be used instead of the content description, if the item is associated with content", + "nullable": false + }, + "use_custom_title": { + "type": "boolean", + "description": "Whether the custom title should be used instead of the content title, if the item is associated with content", + "nullable": false + }, + "use_custom_url": { + "type": "boolean", + "description": "Whether the custom url should be used instead of the content url, if the item is associated with content", + "nullable": false + }, "view_count": { "type": "integer", "format": "int64", "readOnly": true, "description": "Number of times content has been viewed, if present", "nullable": true + }, + "custom_image_data_base64": { + "type": "string", + "x-looker-write-only": true, + "description": "(Write-Only) base64 encoded image data", + "nullable": true + }, + "custom_image_url": { + "type": "string", + "readOnly": true, + "description": "Custom image_url entered by the user, if present", + "nullable": true + }, + "use_custom_image": { + "type": "boolean", + "description": "Whether the custom image should be used instead of the content image, if the item is associated with content", + "nullable": false } }, "x-looker-status": "beta" @@ -34127,8 +33744,7 @@ "nullable": false }, "content_metadata_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of associated content_metadata record", "nullable": true @@ -34161,8 +33777,7 @@ "nullable": true }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "nullable": false @@ -34170,8 +33785,7 @@ "section_order": { "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" }, "description": "ids of the board sections in the order they should be displayed", "nullable": true @@ -34189,8 +33803,7 @@ "nullable": true }, "user_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "User id of board creator", "nullable": true @@ -34234,8 +33847,7 @@ "nullable": true }, "board_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "Id reference to parent board", "nullable": true }, @@ -34249,8 +33861,7 @@ "nullable": true }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "nullable": false @@ -34258,8 +33869,7 @@ "item_order": { "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" }, "description": "ids of the board items in the order they should be displayed", "nullable": true @@ -34267,8 +33877,7 @@ "visible_item_order": { "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" }, "readOnly": true, "description": "ids of the homepage items the user can see in the order they should be displayed", @@ -34403,82 +34012,21 @@ }, "x-looker-status": "stable" }, - "Command": { - "properties": { - "id": { - "type": "integer", - "format": "int64", - "readOnly": true, - "description": "Id of the command record", - "nullable": false - }, - "author_id": { - "type": "integer", - "format": "int64", - "readOnly": true, - "description": "Id of the command author", - "nullable": false - }, - "name": { - "type": "string", - "description": "Name of the command", - "nullable": false - }, - "description": { - "type": "string", - "description": "Description of the command", - "nullable": true - }, - "linked_content_id": { - "type": "string", - "description": "Id of the content associated with the command", - "nullable": false - }, - "linked_content_type": { - "type": "string", - "enum": [ - "dashboard", - "lookml_dashboard" - ], - "description": "Name of the command Valid values are: \"dashboard\", \"lookml_dashboard\".", - "nullable": false - } - }, - "x-looker-status": "beta" - }, - "UpdateCommand": { - "properties": { - "name": { - "type": "string", - "description": "Name of the command", - "nullable": true - }, - "description": { - "type": "string", - "description": "Description of the command", - "nullable": true - } - }, - "x-looker-status": "beta" - }, "ContentFavorite": { "properties": { "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "nullable": false }, "user_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "User Id which owns this ContentFavorite", "nullable": false }, "content_metadata_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "Content Metadata Id associated with this ContentFavorite", "nullable": false }, @@ -34489,8 +34037,7 @@ "nullable": true }, "dashboard_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of a dashboard", "nullable": true @@ -34502,8 +34049,7 @@ "$ref": "#/components/schemas/DashboardBase" }, "board_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of a board", "nullable": true @@ -34545,15 +34091,13 @@ "nullable": true }, "group_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "ID of associated group", "nullable": true }, "user_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "ID of associated user", "nullable": true @@ -34573,8 +34117,7 @@ "nullable": false }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "nullable": false @@ -34586,8 +34129,7 @@ "nullable": true }, "parent_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of Parent Content", "nullable": true @@ -34622,8 +34164,7 @@ "nullable": false }, "inheriting_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of Inherited Content", "nullable": true @@ -34863,8 +34404,7 @@ "nullable": true }, "query_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "Id Of Query", "nullable": true }, @@ -34892,6 +34432,11 @@ "type": "string", "description": "Type", "nullable": true + }, + "rich_content_json": { + "type": "string", + "description": "JSON with all the properties required for rich editor and buttons elements", + "nullable": true } }, "x-looker-status": "stable" @@ -34996,8 +34541,7 @@ "nullable": true }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "nullable": false @@ -35008,8 +34552,7 @@ "ContentValidationAlert": { "properties": { "id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "ID of the alert", "nullable": false }, @@ -35083,8 +34626,7 @@ "nullable": false }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "nullable": false @@ -35096,8 +34638,7 @@ "nullable": true }, "dashboard_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of the viewed Dashboard", "nullable": true @@ -35109,22 +34650,19 @@ "nullable": true }, "content_metadata_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Content metadata id of the Look or Dashboard", "nullable": true }, "user_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of user content was viewed by", "nullable": true }, "group_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of group content was viewed by", "nullable": true @@ -35210,15 +34748,13 @@ "CreateOAuthApplicationUserStateResponse": { "properties": { "user_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "User Id", "nullable": false }, "oauth_application_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "OAuth Application ID", "nullable": false @@ -35242,8 +34778,7 @@ "nullable": false }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "nullable": false @@ -35294,8 +34829,7 @@ "nullable": false }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "nullable": false @@ -35506,8 +35040,7 @@ "nullable": true }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "nullable": false @@ -36057,8 +35590,7 @@ "$ref": "#/components/schemas/Query" }, "query_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "Id Of Query", "nullable": true }, @@ -36078,8 +35610,7 @@ "$ref": "#/components/schemas/ResultMakerWithIdVisConfigAndDynamicFields" }, "result_maker_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "ID of the ResultMakerLookup entry.", "nullable": true }, @@ -36115,6 +35646,11 @@ "description": "Count of Alerts associated to a dashboard element", "nullable": true }, + "rich_content_json": { + "type": "string", + "description": "JSON with all the properties required for rich editor and buttons elements", + "nullable": true + }, "title_text_as_html": { "type": "string", "readOnly": true, @@ -36514,15 +36050,13 @@ "nullable": false }, "content_favorite_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Content Favorite Id", "nullable": true }, "content_metadata_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of content metadata", "nullable": true @@ -36578,8 +36112,7 @@ "nullable": true }, "user_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of User", "nullable": true @@ -36656,8 +36189,7 @@ "nullable": true }, "deleter_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of User that 'soft' deleted the dashboard.", "nullable": true @@ -36703,8 +36235,7 @@ "nullable": true }, "last_updater_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of User that most recently updated the dashboard.", "nullable": true @@ -36734,12 +36265,12 @@ "show_filters_bar": { "type": "boolean", "description": "Show filters bar. **Security Note:** This property only affects the *cosmetic* appearance of the dashboard, not a user's ability to access data. Hiding the filters bar does **NOT** prevent users from changing filters by other means. For information on how to set up secure data access control policies, see [Control User Access to Data](https://looker.com/docs/r/api/control-access)", - "nullable": false + "nullable": true }, "show_title": { "type": "boolean", "description": "Show title", - "nullable": false + "nullable": true }, "folder_id": { "type": "string", @@ -36972,8 +36503,7 @@ "nullable": true }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique ID of the datagroup", "nullable": false @@ -37287,8 +36817,7 @@ "nullable": true }, "oauth_application_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "An External OAuth Application to use for authenticating to the database", "nullable": true }, @@ -37432,8 +36961,7 @@ "nullable": false }, "installation_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Installation ID", "nullable": false @@ -37596,6 +37124,12 @@ "description": "Whether the dialect supports query cost estimates", "nullable": false }, + "cost_estimate_style": { + "type": "string", + "readOnly": true, + "description": "How the dialect handles cost estimation", + "nullable": true + }, "persistent_table_indexes": { "type": "string", "readOnly": true, @@ -37779,8 +37313,7 @@ "group_ids": { "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" }, "description": "List of Looker group ids in which to enroll the embed user", "nullable": true @@ -37800,8 +37333,7 @@ "nullable": true }, "secret_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "Id of the embed secret to use to sign this SSO url. If specified, the value must be an id of a valid (active) secret defined in the Looker instance. If not specified, the URL will be signed with the newest active embed secret defined in the Looker instance.", "nullable": true } @@ -37830,8 +37362,7 @@ "nullable": false }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "nullable": false @@ -37843,8 +37374,7 @@ "nullable": true }, "user_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of user who created this secret", "nullable": true @@ -37921,8 +37451,7 @@ "nullable": false }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "ID of this OAuth Application", "nullable": false @@ -37977,8 +37506,7 @@ "nullable": false }, "content_metadata_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of content metadata", "nullable": true @@ -37991,8 +37519,7 @@ "nullable": true }, "creator_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "User Id of Creator", "nullable": true @@ -38120,8 +37647,7 @@ "nullable": false }, "content_metadata_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of content metadata", "nullable": true @@ -38134,8 +37660,7 @@ "nullable": true }, "creator_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "User Id of Creator", "nullable": true @@ -38429,8 +37954,7 @@ "GroupIdForGroupInclusion": { "properties": { "group_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of group", "nullable": true @@ -38473,8 +37997,7 @@ "nullable": false }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "nullable": false @@ -38535,8 +38058,7 @@ "nullable": false }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "nullable": false @@ -38606,8 +38128,7 @@ "nullable": false }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "nullable": false @@ -38633,8 +38154,7 @@ "parent_group_ids": { "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" }, "readOnly": true, "description": "IDs of parents of this group", @@ -38643,8 +38163,7 @@ "role_ids": { "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" }, "readOnly": true, "description": "Role IDs assigned to group", @@ -38656,8 +38175,7 @@ "GroupIdForGroupUserInclusion": { "properties": { "user_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of user", "nullable": true @@ -38706,8 +38224,7 @@ "nullable": false }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "ID of the hub.", "nullable": false @@ -38786,8 +38303,7 @@ "nullable": false }, "integration_hub_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "ID of the integration hub.", "nullable": false @@ -38850,9 +38366,10 @@ "enum": [ "cell", "query", - "dashboard" + "dashboard", + "none" ], - "description": "A list of action types the integration supports. Valid values are: \"cell\", \"query\", \"dashboard\".", + "description": "A list of action types the integration supports. Valid values are: \"cell\", \"query\", \"dashboard\", \"none\".", "nullable": false }, "supported_formattings": { @@ -38924,8 +38441,7 @@ "installed_delegate_oauth_targets": { "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" }, "description": "Whether the integration is available to users.", "nullable": false @@ -39148,8 +38664,7 @@ "default_new_user_group_ids": { "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" }, "x-looker-write-only": true, "description": "(Write-Only) Array of ids of groups that will be applied to new users the first time they login via LDAP", @@ -39167,8 +38682,7 @@ "default_new_user_role_ids": { "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" }, "x-looker-write-only": true, "description": "(Write-Only) Array of ids of roles that will be applied to new users the first time they login via LDAP", @@ -39425,15 +38939,13 @@ "LDAPGroupRead": { "properties": { "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "nullable": false }, "looker_group_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id of group in Looker", "nullable": true @@ -39472,14 +38984,12 @@ "LDAPGroupWrite": { "properties": { "id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "Unique Id", "nullable": true }, "looker_group_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id of group in Looker", "nullable": true @@ -39497,8 +39007,7 @@ "role_ids": { "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" }, "description": "Looker Role Ids", "nullable": true @@ -39561,8 +39070,7 @@ "user_attribute_ids": { "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" }, "description": "Looker User Attribute Ids", "nullable": true @@ -39794,15 +39302,13 @@ "nullable": false }, "content_metadata_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of content metadata", "nullable": true }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "nullable": false @@ -39814,8 +39320,7 @@ "nullable": true }, "user_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "User Id", "nullable": true } @@ -39834,8 +39339,7 @@ "nullable": false }, "content_metadata_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of content metadata", "nullable": true @@ -39852,14 +39356,12 @@ "nullable": true }, "user_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "User Id", "nullable": true }, "content_favorite_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Content Favorite Id", "nullable": true @@ -39884,8 +39386,7 @@ "nullable": true }, "deleter_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of User that deleted the look.", "nullable": true @@ -39939,8 +39440,7 @@ "nullable": true }, "last_updater_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of User that last updated the look.", "nullable": true @@ -39973,8 +39473,7 @@ "nullable": true }, "query_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "Query Id", "nullable": true }, @@ -40021,8 +39520,7 @@ "nullable": false }, "content_metadata_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of content metadata", "nullable": true @@ -40039,14 +39537,12 @@ "nullable": true }, "user_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "User Id", "nullable": true }, "content_favorite_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Content Favorite Id", "nullable": true @@ -40071,8 +39567,7 @@ "nullable": true }, "deleter_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of User that deleted the look.", "nullable": true @@ -40126,8 +39621,7 @@ "nullable": true }, "last_updater_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of User that last updated the look.", "nullable": true @@ -40160,8 +39654,7 @@ "nullable": true }, "query_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "Query Id", "nullable": true }, @@ -40217,8 +39710,7 @@ "nullable": false }, "content_metadata_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of content metadata", "nullable": true @@ -40235,14 +39727,12 @@ "nullable": true }, "user_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "User Id", "nullable": true }, "content_favorite_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Content Favorite Id", "nullable": true @@ -40267,8 +39757,7 @@ "nullable": true }, "deleter_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of User that deleted the look.", "nullable": true @@ -40322,8 +39811,7 @@ "nullable": true }, "last_updater_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of User that last updated the look.", "nullable": true @@ -40356,8 +39844,7 @@ "nullable": true }, "query_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "Query Id", "nullable": true }, @@ -41802,8 +41289,7 @@ "nullable": true }, "result_maker_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique to get results", "nullable": true @@ -41856,8 +41342,7 @@ "nullable": true }, "query_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "Id of the query to merge", "nullable": true } @@ -41918,8 +41403,7 @@ "nullable": false }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "nullable": false @@ -41984,8 +41468,7 @@ "nullable": false }, "group_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "If set, only Looker users who are members of this group can use this web app with Looker. If group_id is not set, any Looker user may use this app to access this Looker instance", "nullable": true }, @@ -42043,8 +41526,7 @@ "default_new_user_group_ids": { "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" }, "x-looker-write-only": true, "description": "(Write-Only) Array of ids of groups that will be applied to new users the first time they login via OIDC", @@ -42062,8 +41544,7 @@ "default_new_user_role_ids": { "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" }, "x-looker-write-only": true, "description": "(Write-Only) Array of ids of roles that will be applied to new users the first time they login via OIDC", @@ -42123,8 +41604,7 @@ "nullable": true }, "modified_by": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "User id of user who last modified this config", "nullable": true @@ -42230,15 +41710,13 @@ "OIDCGroupRead": { "properties": { "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "nullable": false }, "looker_group_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id of group in Looker", "nullable": true @@ -42270,14 +41748,12 @@ "OIDCGroupWrite": { "properties": { "id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "Unique Id", "nullable": true }, "looker_group_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id of group in Looker", "nullable": true @@ -42295,8 +41771,7 @@ "role_ids": { "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" }, "description": "Looker Role Ids", "nullable": true @@ -42345,8 +41820,7 @@ "user_attribute_ids": { "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" }, "description": "Looker User Attribute Ids", "nullable": true @@ -42443,8 +41917,7 @@ "nullable": false }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "nullable": false @@ -42910,8 +42383,7 @@ "nullable": false }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "nullable": false @@ -43086,8 +42558,7 @@ "nullable": false }, "query_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "Id of query to run", "nullable": true }, @@ -43153,8 +42624,7 @@ "nullable": false }, "query_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "Id of query", "nullable": true }, @@ -43268,8 +42738,7 @@ "nullable": true }, "dashboard_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of dashboard to render", "nullable": true @@ -43312,8 +42781,7 @@ "nullable": true }, "query_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of query to render", "nullable": true @@ -43364,8 +42832,7 @@ "nullable": true }, "user_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "The user account permissions in which the render task will execute", "nullable": true @@ -43499,8 +42966,7 @@ "ResultMakerWithIdVisConfigAndDynamicFields": { "properties": { "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id.", "nullable": false @@ -43542,8 +43008,7 @@ "nullable": false }, "query_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "ID of query if this is a query.", "nullable": true @@ -43582,8 +43047,7 @@ "nullable": false }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "nullable": false @@ -43597,8 +43061,7 @@ "$ref": "#/components/schemas/PermissionSet" }, "permission_set_id": { - "type": "integer", - "format": "int64", + "type": "string", "x-looker-write-only": true, "description": "(Write-Only) Id of permission set", "nullable": true @@ -43607,8 +43070,7 @@ "$ref": "#/components/schemas/ModelSet" }, "model_set_id": { - "type": "integer", - "format": "int64", + "type": "string", "x-looker-write-only": true, "description": "(Write-Only) Id of model set", "nullable": true @@ -43642,8 +43104,7 @@ "nullable": false }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "nullable": false @@ -43657,8 +43118,7 @@ "$ref": "#/components/schemas/PermissionSet" }, "permission_set_id": { - "type": "integer", - "format": "int64", + "type": "string", "x-looker-write-only": true, "description": "(Write-Only) Id of permission set", "nullable": true @@ -43667,8 +43127,7 @@ "$ref": "#/components/schemas/ModelSet" }, "model_set_id": { - "type": "integer", - "format": "int64", + "type": "string", "x-looker-write-only": true, "description": "(Write-Only) Id of model set", "nullable": true @@ -43709,8 +43168,7 @@ "nullable": false }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "nullable": false @@ -43927,8 +43385,7 @@ "default_new_user_role_ids": { "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" }, "x-looker-write-only": true, "description": "(Write-Only) Array of ids of roles that will be applied to new users the first time they login via Saml", @@ -43937,8 +43394,7 @@ "default_new_user_group_ids": { "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" }, "x-looker-write-only": true, "description": "(Write-Only) Array of ids of groups that will be applied to new users the first time they login via Saml", @@ -44036,15 +43492,13 @@ "SamlGroupRead": { "properties": { "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "nullable": false }, "looker_group_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id of group in Looker", "nullable": true @@ -44083,14 +43537,12 @@ "SamlGroupWrite": { "properties": { "id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "Unique Id", "nullable": true }, "looker_group_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id of group in Looker", "nullable": true @@ -44108,8 +43560,7 @@ "role_ids": { "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" }, "description": "Looker Role Ids", "nullable": true @@ -44204,8 +43655,7 @@ "user_attribute_ids": { "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" }, "description": "Looker User Attribute Ids", "nullable": true @@ -44223,15 +43673,13 @@ "ScheduledPlanDestination": { "properties": { "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "nullable": false }, "scheduled_plan_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "Id of a scheduled plan you own", "nullable": true }, @@ -44293,8 +43741,7 @@ "nullable": true }, "user_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "User Id which owns this scheduled plan", "nullable": true }, @@ -44314,8 +43761,7 @@ "nullable": true }, "dashboard_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "Id of a dashboard", "nullable": true }, @@ -44435,8 +43881,7 @@ "nullable": true }, "user_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "User Id which owns this scheduled plan", "nullable": true }, @@ -44456,8 +43901,7 @@ "nullable": true }, "dashboard_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "Id of a dashboard", "nullable": true }, @@ -44567,8 +44011,7 @@ "nullable": true }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "nullable": false @@ -45121,8 +44564,7 @@ "nullable": false }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "nullable": false @@ -45183,8 +44625,7 @@ "nullable": true }, "sudo_user_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Actual user in the case when this session represents one user sudo'ing as another", "nullable": true @@ -45228,8 +44669,8 @@ "description": "Toggle marketplace on or off", "nullable": false }, - "whitelabel_configuration": { - "$ref": "#/components/schemas/WhitelabelConfiguration" + "privatelabel_configuration": { + "$ref": "#/components/schemas/PrivatelabelConfiguration" }, "custom_welcome_email": { "$ref": "#/components/schemas/CustomWelcomeEmail" @@ -45434,8 +44875,7 @@ "nullable": true }, "result_maker_id": { - "type": "integer", - "format": "int64", + "type": "string", "description": "ID of the ResultMakerLookup entry.", "nullable": true } @@ -45778,8 +45218,7 @@ "nullable": true }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "nullable": false @@ -45830,8 +45269,7 @@ "nullable": false }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "nullable": false @@ -45908,22 +45346,19 @@ "nullable": false }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id of this group-attribute relation", "nullable": false }, "group_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of group", "nullable": true }, "user_attribute_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of user attribute", "nullable": true @@ -45986,8 +45421,7 @@ "nullable": true }, "user_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of User", "nullable": true @@ -46005,8 +45439,7 @@ "nullable": false }, "user_attribute_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Id of User Attribute", "nullable": true @@ -46069,8 +45502,7 @@ "nullable": true }, "user_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "User ID", "nullable": true @@ -46187,10 +45619,10 @@ "nullable": true }, "embed_group_space_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, - "description": "(Embed only) ID of user's group space based on the external_group_id optionally specified during embed user login", + "x-looker-deprecated": true, + "description": "(DEPRECATED) (Embed only) ID of user's group space based on the external_group_id optionally specified during embed user login", "nullable": true }, "first_name": { @@ -46201,8 +45633,7 @@ "group_ids": { "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" }, "readOnly": true, "description": "Array of ids of the groups for this user", @@ -46214,8 +45645,7 @@ "nullable": true }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "nullable": false @@ -46250,8 +45680,7 @@ "nullable": true }, "personal_folder_id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "ID of user's personal folder", "nullable": true @@ -46265,8 +45694,7 @@ "role_ids": { "type": "array", "items": { - "type": "integer", - "format": "int64" + "type": "string" }, "readOnly": true, "description": "Array of ids of the roles for this user", @@ -46319,6 +45747,12 @@ "description": "User can inherit roles from a normal Looker group.", "nullable": false }, + "embed_group_folder_id": { + "type": "string", + "readOnly": true, + "description": "(Embed only) ID of user's group folder based on the external_group_id optionally specified during embed user login", + "nullable": true + }, "url": { "type": "string", "format": "uri-reference", @@ -46341,8 +45775,7 @@ "nullable": false }, "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "nullable": false @@ -46470,8 +45903,7 @@ "WhitelabelConfiguration": { "properties": { "id": { - "type": "integer", - "format": "int64", + "type": "string", "readOnly": true, "description": "Unique Id", "nullable": false @@ -46556,6 +45988,88 @@ }, "x-looker-status": "stable" }, + "PrivatelabelConfiguration": { + "properties": { + "logo_file": { + "type": "string", + "description": "Customer logo image. Expected base64 encoded data (write-only)", + "nullable": true + }, + "logo_url": { + "type": "string", + "readOnly": true, + "description": "Logo image url (read-only)", + "nullable": true + }, + "favicon_file": { + "type": "string", + "description": "Custom favicon image. Expected base64 encoded data (write-only)", + "nullable": true + }, + "favicon_url": { + "type": "string", + "readOnly": true, + "description": "Favicon image url (read-only)", + "nullable": true + }, + "default_title": { + "type": "string", + "description": "Default page title", + "nullable": true + }, + "show_help_menu": { + "type": "boolean", + "description": "Boolean to toggle showing help menus", + "nullable": false + }, + "show_docs": { + "type": "boolean", + "description": "Boolean to toggle showing docs", + "nullable": false + }, + "show_email_sub_options": { + "type": "boolean", + "description": "Boolean to toggle showing email subscription options.", + "nullable": false + }, + "allow_looker_mentions": { + "type": "boolean", + "description": "Boolean to toggle mentions of Looker in emails", + "nullable": false + }, + "allow_looker_links": { + "type": "boolean", + "description": "Boolean to toggle links to Looker in emails", + "nullable": false + }, + "custom_welcome_email_advanced": { + "type": "boolean", + "description": "Allow subject line and email heading customization in customized emails”", + "nullable": false + }, + "setup_mentions": { + "type": "boolean", + "description": "Remove the word Looker from appearing in the account setup page", + "nullable": false + }, + "alerts_logo": { + "type": "boolean", + "description": "Remove Looker logo from Alerts", + "nullable": false + }, + "alerts_links": { + "type": "boolean", + "description": "Remove Looker links from Alerts", + "nullable": false + }, + "folders_mentions": { + "type": "boolean", + "description": "Remove Looker mentions in home folder page when you don’t have any items saved", + "nullable": false + } + }, + "x-looker-status": "stable" + }, "Workspace": { "properties": { "can": {