blob: 22c70208b98bcb6d19d7e56745133633cef8ce13 [file]
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you 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
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*/
import { TypeInfo } from "../typeInfo";
import { CodecBuilder } from "./builder";
import { BaseSerializerGenerator } from "./serializer";
import { CodegenRegistry } from "./router";
import { TypeId, MaxUInt32 } from "../type";
import { Scope } from "./scope";
class EnumSerializerGenerator extends BaseSerializerGenerator {
typeInfo: TypeInfo;
constructor(typeInfo: TypeInfo, builder: CodecBuilder, scope: Scope) {
super(typeInfo, builder, scope);
this.typeInfo = typeInfo;
}
private getEnumEntries(): Array<[string, string | number]> {
const enumProps = this.typeInfo.options?.enumProps;
if (!enumProps) {
return [];
}
return Object.entries(enumProps).filter(([key, value]) => {
return !(typeof value === "string" && Number.isInteger(Number(key)));
});
}
private useExplicitNumericWireValues(entries: Array<[string, string | number]>): boolean {
if (entries.length < 1) {
throw new Error("An enum must contain at least one field");
}
const seen = new Set<number>();
for (const [, value] of entries) {
if (typeof value === "string") {
return false;
}
if (!Number.isInteger(value) || value > MaxUInt32 || value < 0) {
throw new Error("Enum value must be a valid uint32");
}
if (seen.has(value)) {
throw new Error("Enum numeric values must be unique");
}
seen.add(value);
}
return true;
}
write(accessor: string): string {
if (!this.typeInfo.options?.enumProps) {
return this.builder.writer.writeVarUInt32(accessor);
}
const enumEntries = this.getEnumEntries();
const useExplicitNumericWireValues = this.useExplicitNumericWireValues(enumEntries);
return `
${enumEntries.map(([, value], index) => {
if (typeof value !== "string" && typeof value !== "number") {
throw new Error("Enum value must be string or number");
}
if (typeof value === "number") {
if (value > MaxUInt32 || value < 0) {
throw new Error("Enum value must be a valid uint32");
}
}
const safeValue = typeof value === "string" ? `"${value}"` : value;
const wireValue = useExplicitNumericWireValues ? safeValue : index;
return ` if (${accessor} === ${safeValue}) {
${this.builder.writer.writeVarUInt32(wireValue)}
}`;
}).join(" else ")}
else {
throw new Error("Enum received an unexpected value: " + ${accessor});
}
`;
}
readTypeInfo(): string {
const internalTypeId = this.getInternalTypeId();
let readUserTypeIdStmt = "";
let namesStmt = "";
switch (internalTypeId) {
case TypeId.ENUM:
readUserTypeIdStmt = `${this.builder.reader.readVarUint32Small7()};`;
break;
case TypeId.NAMED_ENUM:
namesStmt = `
${
// skip the namespace
this.builder.metaStringResolver.readNamespace()
}
${
// skip the namespace
this.builder.metaStringResolver.readTypeName()
}
`;
break;
default:
break;
}
return `
${
// skip the typeId
this.builder.reader.readUint8()
}
${readUserTypeIdStmt}
${namesStmt}
`;
}
writeTypeInfo(): string {
const internalTypeId = this.getInternalTypeId();
let typeMeta = "";
let writeUserTypeIdStmt = "";
switch (internalTypeId) {
case TypeId.ENUM:
writeUserTypeIdStmt = this.builder.writer.writeVarUint32Small7(this.typeInfo.userTypeId);
break;
case TypeId.NAMED_ENUM:
{
const typeInfo = this.typeInfo;
const nsBytes = this.scope.declare("nsBytes", this.builder.metaStringResolver.encodeNamespace(CodecBuilder.replaceBackslashAndQuote(typeInfo.namespace)));
const typeNameBytes = this.scope.declare("typeNameBytes", this.builder.metaStringResolver.encodeTypeName(CodecBuilder.replaceBackslashAndQuote(typeInfo.typeName)));
typeMeta = `
${this.builder.metaStringResolver.writeBytes(nsBytes)}
${this.builder.metaStringResolver.writeBytes(typeNameBytes)}
`;
}
break;
default:
break;
}
return `
${this.builder.writer.writeUint8(this.getTypeId())};
${writeUserTypeIdStmt}
${typeMeta}
`;
}
read(accessor: (expr: string) => string): string {
if (!this.typeInfo.options?.enumProps) {
return accessor(this.builder.reader.readVarUInt32());
}
const enumEntries = this.getEnumEntries();
const useExplicitNumericWireValues = this.useExplicitNumericWireValues(enumEntries);
const enumValue = this.scope.uniqueName("enum_v");
return `
const ${enumValue} = ${this.builder.reader.readVarUInt32()};
switch(${enumValue}) {
${enumEntries.map(([, value], index) => {
if (typeof value !== "string" && typeof value !== "number") {
throw new Error("Enum value must be string or number");
}
if (typeof value === "number") {
if (value > MaxUInt32 || value < 0) {
throw new Error("Enum value must be a valid uint32");
}
}
const safeValue = typeof value === "string" ? `"${value}"` : `${value}`;
const wireValue = useExplicitNumericWireValues ? safeValue : `${index}`;
return `
case ${wireValue}:
${accessor(safeValue)}
break;
`;
}).join("\n")}
default:
throw new Error("Enum received an unexpected value: " + ${enumValue});
}
`;
}
getFixedSize(): number {
return 7;
}
}
CodegenRegistry.register(TypeId.ENUM, EnumSerializerGenerator);
CodegenRegistry.register(TypeId.NAMED_ENUM, EnumSerializerGenerator);