Skip to content

Commit

Permalink
wip
Browse files Browse the repository at this point in the history
  • Loading branch information
scobb committed Dec 18, 2023
1 parent e3ae0a3 commit cd54e99
Show file tree
Hide file tree
Showing 3 changed files with 211 additions and 189 deletions.
352 changes: 168 additions & 184 deletions src/Codegen/Constraints/BaseBuilder.php
Original file line number Diff line number Diff line change
Expand Up @@ -3,191 +3,175 @@
namespace Slack\Hack\JsonSchema\Codegen;

use namespace HH\Lib\{C, Str};
use type Facebook\HackCodegen\{
CodegenClass,
// CodegenEnum,
CodegenEnumMember,
CodegenMethod,
CodegenProperty,
HackBuilder,
HackBuilderValues,
};
use type Facebook\HackCodegen\{CodegenClass, CodegenMethod, CodegenProperty, HackBuilder, HackBuilderValues};

<<__ConsistentConstruct>>
abstract class BaseBuilder<T> implements IBuilder {
use Factory;

protected T $typed_schema;
protected static string $schema_name = '';

public function __construct(
protected Context $ctx,
protected string $suffix,
protected TSchema $schema,
protected ?CodegenClass $class = null,
) {
$this->typed_schema = type_assert_shape($this->schema, static::$schema_name);
}

/**
*
* Return a string which will get rendered as a literal value describing the
* output type of the schema. For example, the `StringBuilder` would return:
* `return 'string';`, the `NumberBuilder` would return: `return 'num';`.
*/
abstract public function getType(): string;

public function isArrayKeyType(): bool {
$type = $this->getType();
if ($type === 'string' || $type === 'int') {
return true;
}

$schema = type_assert_type($this->typed_schema, TSchema::class);
return Shapes::keyExists($schema, 'hackEnum');
}

/**
*
* Main method for building the class for the schema and appending it to the
* file.
*/
abstract public function build(): this;

protected function getHackBuilder(): HackBuilder {
return new HackBuilder($this->ctx->getHackCodegenFactory()->getConfig());
}

public function getClassName(): string {
return $this->generateClassName($this->ctx->getClassName(), $this->suffix);
}

protected function codegenClass(): CodegenClass {
if ($this->class) {
return $this->class;
}

return $this->ctx
->getHackCodegenFactory()
->codegenClass($this->getClassName())
->setIsFinal(true);
}

protected function codegenProperty(string $name): CodegenProperty {
return $this->ctx
->getHackCodegenFactory()
->codegenProperty($name)
->setIsStatic(true)
->setPrivate();
}

protected function codegenCheckMethod(): CodegenMethod {
return $this->ctx
->getHackCodegenFactory()
->codegenMethod('check')
->setPublic()
->setIsStatic(true);
}

protected function getEnumCodegenProperty(): ?CodegenProperty {
$property = null;
$schema = type_assert_shape($this->typed_schema, 'Slack\Hack\JsonSchema\Codegen\TSchema');
$enum = $schema['enum'] ?? null;
if ($enum is nonnull) {
$should_create_hack_enum = $schema['generateHackEnum'] ?? false;
if ($should_create_hack_enum) {
$factory = $this->ctx->getHackCodegenFactory()
$members = Vec\map($enum, $val ==> $factory->codegenEnumMember($val));
$enum_obj = $factory->codegenEnum('enum', 'string')->addMembers($members);
$this->codegenProperty('enum')->setType('enum')->setPublic()->setValue($enum_obj->getCode(), HackBuilderValues::literal());
} else {
$hb = $this->getHackBuilder()
->addValue($enum, HackBuilderValues::vec(HackBuilderValues::export()));

$property = $this->codegenProperty('enum')
->setType('vec<mixed>')
->setValue($hb->getCode(), HackBuilderValues::literal());
}
}
return $property;
}

protected function addEnumConstraintCheck(HackBuilder $hb): void {
$schema = type_assert_shape($this->typed_schema, 'Slack\Hack\JsonSchema\Codegen\TSchema');
if (($schema['enum'] ?? null) is nonnull) {
$hb->addMultilineCall('Constraints\EnumConstraint::check', vec['$typed', 'self::$enum', '$pointer']);
}
}

protected function addHackEnumConstraintCheck(HackBuilder $hb): void {
$schema = type_assert_type($this->typed_schema, TSchema::class);
if (!Shapes::keyExists($schema, 'hackEnum')) {
return;
}

try {
$rc = new \ReflectionClass($schema['hackEnum']);
} catch (\ReflectionException $e) {
throw new \Exception(Str\format("Hack enum '%s' does not exist", $schema['hackEnum']));
}

invariant($rc->isEnum(), "'%s' is not an enum", $schema['hackEnum']);

$schema_type = $schema['type'] ?? null;
$hack_enum_values = keyset[];
foreach ($rc->getConstants() as $hack_enum_value) {
if ($schema_type === TSchemaType::INTEGER_T) {
$hack_enum_value = $hack_enum_value ?as int;
} else {
$hack_enum_value = $hack_enum_value ?as string;
}
invariant(
$hack_enum_value is nonnull,
"'%s' must contain only values of type %s",
$rc->getName(),
$schema_type === TSchemaType::INTEGER_T ? 'int' : 'string',
);
$hack_enum_values[] = $hack_enum_value;
}

if (Shapes::keyExists($schema, 'enum')) {
// If both `enum` and `hackEnum` are specified, assert that `enum` is a subset of
// `hackEnum` values. Any value not also in `hackEnum` can't be valid.
foreach ($schema['enum'] as $enum_value) {
invariant(
$enum_value is string,
"Enum value '%s' is not a valid value for '%s'",
\print_r($enum_value, true),
$rc->getName(),
);
invariant(
C\contains_key($hack_enum_values, $enum_value),
"Enum value '%s' is unexpectedly not present in '%s'",
\print_r($enum_value, true),
$rc->getName(),
);
}
}

$hb->addMultilineCall(
'$typed = Constraints\HackEnumConstraint::check',
vec[
'$typed',
Str\format('\%s::class', $rc->getName()),
'$pointer',
],
);
}

public function addBuilderClass(CodegenClass $class): void {
if ($this->class) {
return;
}

$this->ctx->getFile()->addClass($class);
}

public function setSuffix(string $suffix): void {
$this->suffix = $suffix;
}
use Factory;

protected T $typed_schema;
protected static string $schema_name = '';

public function __construct(
protected Context $ctx,
protected string $suffix,
protected TSchema $schema,
protected ?CodegenClass $class = null,
) {
$this->typed_schema = type_assert_shape($this->schema, static::$schema_name);
}

/**
*
* Return a string which will get rendered as a literal value describing the
* output type of the schema. For example, the `StringBuilder` would return:
* `return 'string';`, the `NumberBuilder` would return: `return 'num';`.
*/
abstract public function getType(): string;

public function isArrayKeyType(): bool {
$type = $this->getType();
if ($type === 'string' || $type === 'int') {
return true;
}

$schema = type_assert_type($this->typed_schema, TSchema::class);
return Shapes::keyExists($schema, 'hackEnum');
}

/**
*
* Main method for building the class for the schema and appending it to the
* file.
*/
abstract public function build(): this;

protected function getHackBuilder(): HackBuilder {
return new HackBuilder($this->ctx->getHackCodegenFactory()->getConfig());
}

public function getClassName(): string {
return $this->generateClassName($this->ctx->getClassName(), $this->suffix);
}

protected function codegenClass(): CodegenClass {
if ($this->class) {
return $this->class;
}

return $this->ctx
->getHackCodegenFactory()
->codegenClass($this->getClassName())
->setIsFinal(true);
}

protected function codegenProperty(string $name): CodegenProperty {
return $this->ctx
->getHackCodegenFactory()
->codegenProperty($name)
->setIsStatic(true)
->setPrivate();
}

protected function codegenCheckMethod(): CodegenMethod {
return $this->ctx
->getHackCodegenFactory()
->codegenMethod('check')
->setPublic()
->setIsStatic(true);
}

protected function getEnumCodegenProperty(): ?CodegenProperty {
$property = null;
$schema = type_assert_shape($this->typed_schema, 'Slack\Hack\JsonSchema\Codegen\TSchema');
$enum = $schema['enum'] ?? null;
if ($enum is nonnull) {
$hb = $this->getHackBuilder()
->addValue($enum, HackBuilderValues::vec(HackBuilderValues::export()));

$property = $this->codegenProperty('enum')
->setType('vec<mixed>')
->setValue($hb->getCode(), HackBuilderValues::literal());
}
return $property;
}

protected function addEnumConstraintCheck(HackBuilder $hb): void {
$schema = type_assert_shape($this->typed_schema, 'Slack\Hack\JsonSchema\Codegen\TSchema');
if (($schema['enum'] ?? null) is nonnull) {
$hb->addMultilineCall('Constraints\EnumConstraint::check', vec['$typed', 'self::$enum', '$pointer']);
}
}

protected function addHackEnumConstraintCheck(HackBuilder $hb): void {
$schema = type_assert_type($this->typed_schema, TSchema::class);
if (!Shapes::keyExists($schema, 'hackEnum')) {
return;
}

try {
$rc = new \ReflectionClass($schema['hackEnum']);
} catch (\ReflectionException $e) {
throw new \Exception(Str\format("Hack enum '%s' does not exist", $schema['hackEnum']));
}

invariant($rc->isEnum(), "'%s' is not an enum", $schema['hackEnum']);

$schema_type = $schema['type'] ?? null;
$hack_enum_values = keyset[];
foreach ($rc->getConstants() as $hack_enum_value) {
if ($schema_type === TSchemaType::INTEGER_T) {
$hack_enum_value = $hack_enum_value ?as int;
} else {
$hack_enum_value = $hack_enum_value ?as string;
}
invariant(
$hack_enum_value is nonnull,
"'%s' must contain only values of type %s",
$rc->getName(),
$schema_type === TSchemaType::INTEGER_T ? 'int' : 'string',
);
$hack_enum_values[] = $hack_enum_value;
}

if (Shapes::keyExists($schema, 'enum')) {
// If both `enum` and `hackEnum` are specified, assert that `enum` is a subset of
// `hackEnum` values. Any value not also in `hackEnum` can't be valid.
foreach ($schema['enum'] as $enum_value) {
invariant(
$enum_value is string,
"Enum value '%s' is not a valid value for '%s'",
\print_r($enum_value, true),
$rc->getName(),
);
invariant(
C\contains_key($hack_enum_values, $enum_value),
"Enum value '%s' is unexpectedly not present in '%s'",
\print_r($enum_value, true),
$rc->getName(),
);
}
}

$hb->addMultilineCall(
'$typed = Constraints\HackEnumConstraint::check',
vec[
'$typed',
Str\format('\%s::class', $rc->getName()),
'$pointer',
],
);
}

public function addBuilderClass(CodegenClass $class): void {
if ($this->class) {
return;
}

$this->ctx->getFile()->addClass($class);
}

public function setSuffix(string $suffix): void {
$this->suffix = $suffix;
}
}
Loading

0 comments on commit cd54e99

Please sign in to comment.