File Coverage

blib/lib/DBIx/Result/Convert/JSONSchema.pm
Criterion Covered Total %
statement 100 136 73.5
branch 47 72 65.2
condition 43 67 64.1
subroutine 9 10 90.0
pod 1 1 100.0
total 200 286 69.9


line stmt bran cond sub pod time code
1             package DBIx::Result::Convert::JSONSchema;
2              
3             our $VERSION = '0.06';
4              
5              
6             =head1 NAME
7              
8             DBIx::Result::Convert::JSONSchema - Convert DBIx result schema to JSON schema
9              
10             =begin html
11              
12             Build Status
13             Coverage Status
14              
15             =end html
16              
17             =head1 VERSION
18              
19             0.06
20              
21             =head1 SYNOPSIS
22              
23             use DBIx::Result::Convert::JSONSchema;
24              
25             my $SchemaConvert = DBIx::Result::Convert::JSONSchema->new( schema => Schema );
26             my $json_schema = $SchemaConvert->get_json_schema( DBIx::Class::ResultSource );
27              
28             =head1 DESCRIPTION
29              
30             This module attempts basic conversion of L to equivalent
31             of L.
32             By default the conversion assumes that the L originated
33             from MySQL database. Thus all the types and defaults are set based on MySQL
34             field definitions.
35             It is, however, possible to overwrite field type map and length map to support
36             L from other database solutions.
37              
38             Note, relations between tables are not taken in account!
39              
40             =cut
41              
42              
43 4     4   5101 use Moo;
  4         38893  
  4         17  
44 4     4   6988 use Types::Standard qw/ InstanceOf Enum HashRef /;
  4         260505  
  4         37  
45              
46 4     4   3659 use Carp;
  4         8  
  4         242  
47 4     4   1662 use Module::Load qw/ load /;
  4         3628  
  4         24  
48              
49              
50             has schema => (
51             is => 'ro',
52             isa => InstanceOf['DBIx::Class::Schema'],
53             required => 1,
54             );
55              
56             has schema_source => (
57             is => 'lazy',
58             isa => Enum[ qw/ MySQL / ],
59             default => 'MySQL',
60             );
61              
62             has length_type_map => (
63             is => 'rw',
64             isa => HashRef,
65             default => sub {
66             return {
67             string => [ qw/ minLength maxLength / ],
68             number => [ qw/ minimum maximum / ],
69             integer => [ qw/ minimum maximum / ],
70             };
71             },
72             );
73              
74             has type_map => (
75             is => 'rw',
76             isa => HashRef,
77             default => sub {
78             my ( $self ) = @_;
79              
80             my $type_class = __PACKAGE__ . '::Type::' . $self->schema_source;
81             load $type_class;
82              
83             return $type_class->get_type_map;
84             },
85             );
86              
87             has length_map => (
88             is => 'rw',
89             isa => HashRef,
90             default => sub {
91             my ( $self ) = @_;
92              
93             my $defaults_class = __PACKAGE__ . '::Default::' . $self->schema_source;
94             load $defaults_class;
95              
96             return $defaults_class->get_length_map;
97             },
98             );
99              
100             has pattern_map => (
101             is => 'rw',
102             isa => HashRef,
103             lazy => 1,
104             default => sub {
105             return {
106             time => '^\d{2}:\d{2}:\d{2}$',
107             year => '^\d{4}$',
108             datetime => '^\d{4}-\d{2}-\d{2} \d{2}:\d{2}:\d{2}$',
109             timestamp => '^\d{4}-\d{2}-\d{2} \d{2}:\d{2}:\d{2}$',
110             };
111             }
112             );
113              
114             has format_map => (
115             is => 'rw',
116             isa => HashRef,
117             lazy => 1,
118             default => sub {
119             return {
120             date => 'date',
121             };
122             }
123             );
124              
125              
126             =head2 get_json_schema
127              
128             Returns somewhat equivalent JSON schema based on DBIx result source name.
129              
130             my $json_schema = $converted->get_json_schema( 'TableSource', {
131             schema_declaration => 'http://json-schema.org/draft-04/schema#',
132             decimals_to_pattern => 1,
133             has_schema_property_description => 1,
134             allow_additional_properties => 0,
135             ignore_property_defaults => 1,
136             overwrite_schema_property_keys => {
137             name => 'cat',
138             address => 'dog',
139             },
140             add_schema_properties => {
141             address => { ... },
142             bank_account => '#/definitions/bank_account',
143             },
144             overwrite_schema_properties => {
145             name => {
146             _action => 'merge', # one of - merge/overwrite
147             minimum => 10,
148             maximum => 20,
149             type => 'number',
150             },
151             },
152             include_required => [ qw/ street city / ],
153             exclude_required => [ qw/ name address / ],
154             exclude_properties => [ qw/ mouse house / ],
155              
156             dependencies => {
157             first_name => [ qw/ middle_name last_name / ],
158             },
159             });
160              
161             Optional arguments to change how JSON schema is generated:
162              
163             =over 8
164              
165             =item * schema_declaration
166              
167             Declare which version of the JSON Schema standard that the schema was written against.
168              
169             L
170              
171             B: "http://json-schema.org/schema#"
172              
173             =item * decimals_to_pattern
174              
175             1/0 - value to indicate if 'number' type field should be converted to 'string' type with
176             RegExp pattern based on decimal place definition in database.
177              
178             B: 0
179              
180             =item * has_schema_property_description
181              
182             Generate schema description for fields e.g. 'Optional numeric type value for field context e.g. 1'.
183              
184             B: 0
185              
186             =item * ignore_property_defaults
187              
188             Do not set schema B property field based on default in DBIx schema
189              
190             B: 0
191              
192             =item * allow_additional_properties
193              
194             Define if the schema accepts additional keys in given payload.
195              
196             B: 0
197              
198             =item * add_property_minimum_value
199              
200             If field does not have format type add minimum values for number and string types based on DB field type.
201             This might not make sense in most cases as the minimum is either 0 or the lower bound if number is signed.
202              
203             B: 0
204              
205             =item * overwrite_schema_property_keys
206              
207             HashRef representing mapping between old property name and new property name to overwrite existing schema keys,
208             Properties from old key will be assigned to the new property.
209              
210             B The key conversion is executed last, every other option e.g. C will work only on original
211             database column names.
212              
213             =item * overwrite_schema_properties
214              
215             HashRef of property name and new attributes which can be either overwritten or merged based on given B<_action> key.
216              
217             =item * exclude_required
218              
219             ArrayRef of database column names which should always be EXCLUDED from REQUIRED schema properties.
220              
221             =item * include_required
222              
223             ArrayRef of database column names which should always be INCLUDED in REQUIRED schema properties
224              
225             =item * exclude_properties
226              
227             ArrayRef of database column names which should be excluded from JSON schema AT ALL
228              
229             =item * dependencies
230              
231             L
232              
233             =item * add_schema_properties
234              
235             HashRef of custom schema properties that must be included in final definition
236             Note that custom properties will overwrite defaults
237              
238             =item * schema_overwrite
239              
240             HashRef of top level schema properties e.g. 'required', 'properties' etc. to overwrite
241              
242             =back
243              
244             =cut
245              
246             sub get_json_schema {
247 9     9 1 79174 my ( $self, $source, $args ) = @_;
248              
249 9 100       47 croak 'missing schema source' unless $source;
250              
251 8   100     25 $args //= {};
252              
253             # additional schema generation options
254 8         19 my $decimals_to_pattern = $args->{decimals_to_pattern};
255 8         15 my $has_schema_property_description = $args->{has_schema_property_description};
256 8         14 my $ignore_property_defaults = $args->{ignore_property_defaults};
257 8   100     39 my $overwrite_schema_property_keys = $args->{overwrite_schema_property_keys} // {};
258 8         20 my $add_schema_properties = $args->{add_schema_properties};
259 8   100     30 my $overwrite_schema_properties = $args->{overwrite_schema_properties} // {};
260 8         16 my $add_property_minimum_value = $args->{add_property_minimum_value};
261 8 100       18 my %exclude_required = map { $_ => 1 } @{ $args->{exclude_required} || [] };
  1         5  
  8         40  
262 8 100       17 my %include_required = map { $_ => 1 } @{ $args->{include_required} || [] };
  2         6  
  8         30  
263 8 100       14 my %exclude_properties = map { $_ => 1 } @{ $args->{exclude_properties} || [] };
  1         3  
  8         39  
264              
265 8         55 my $dependencies = $args->{dependencies};
266 8   100     45 my $schema_declaration = $args->{schema_declaration} // 'http://json-schema.org/schema#';
267 8   100     30 my $allow_additional_properties = $args->{allow_additional_properties} // 0;
268 8   100     39 my $schema_overwrite = $args->{schema_overwrite} // {};
269              
270 8 100       70 my %json_schema = (
271             '$schema' => $schema_declaration,
272             type => 'object',
273             required => [],
274             properties => {},
275             additionalProperties => $allow_additional_properties,
276              
277             ( $dependencies ? ( dependencies => $dependencies ) : () ),
278             );
279              
280 8         38 my $source_info = $self->_get_column_info( $source );
281              
282             SCHEMA_COLUMN:
283 7         984 foreach my $column ( keys %{ $source_info } ) {
  7         38  
284 196 100       338 next SCHEMA_COLUMN if $exclude_properties{ $column };
285              
286 195         266 my $column_info = $source_info->{ $column };
287              
288             # DBIx schema data type -> JSON schema data type
289             my $json_type = $self->type_map->{ $column_info->{data_type} }
290             or croak sprintf(
291             'unknown data type - %s (source: %s, column: %s)',
292 195 50       2546 $column_info->{data_type}, $source, $column
293             );
294              
295 195         1227 $json_schema{properties}->{ $column }->{type} = $json_type;
296              
297             # DBIx schema type -> JSON format
298 195         2466 my $format_type = $self->format_map->{ $column_info->{data_type} };
299 195 100       1092 if ( $format_type ) {
300 7         27 $json_schema{properties}->{ $column }->{format} = $format_type;
301             }
302              
303             # DBIx schema size constraint -> JSON schema size constraint
304 195 100 100     2366 if ( ! $format_type && $self->length_map->{ $column_info->{data_type} } ) {
305 139         1507 $self->_set_json_schema_property_range( \%json_schema, $column_info, $column, $add_property_minimum_value );
306             }
307              
308             # DBIx schema required -> JSON schema required
309 195         751 my $is_required_field = $include_required{ $column };
310 195 100 100     808 if ( $is_required_field || ( ! $column_info->{default_value} && ! $column_info->{is_nullable} && ! $exclude_required{ $column } ) ) {
      100        
311 8   33     39 my $required_property = $overwrite_schema_property_keys->{ $column } // $column;
312 8         15 push @{ $json_schema{required} }, $required_property;
  8         23  
313             }
314              
315             # DBIx schema defaults -> JSON schema defaults (no refs e.g. current_timestamp)
316 195 100 100     499 if ( ! $ignore_property_defaults && defined $column_info->{default_value} && ! ref $column_info->{default_value} ) {
      100        
317 6         16 $json_schema{properties}->{ $column }->{default} = $column_info->{default_value};
318             }
319              
320             # DBIx schema list -> JSON enum list
321 195 50 66     391 if ( $json_type eq 'enum' && $column_info->{extra} && $column_info->{extra}->{list} ) { # no autovivification
      33        
322 14         48 $json_schema{properties}->{ $column }->{enum} = $column_info->{extra}->{list};
323             }
324              
325             # Consider 'is nullable' to accept 'null' values in all cases where field is not explicitly required
326 195 100 100     536 if ( ! $is_required_field && $column_info->{is_nullable} ) {
327 172 100       278 if ( $json_type eq 'enum' ) {
328 14   50     33 $json_schema{properties}->{ $column }->{enum} //= [];
329 14         19 push @{ $json_schema{properties}->{ $column }->{enum} }, 'null';
  14         53  
330             }
331             else {
332 158         343 $json_schema{properties}->{ $column }->{type} = [ $json_type, 'null' ];
333             }
334             }
335              
336             # DBIx decimal numbers -> JSON schema numeric string pattern
337 195 100 100     402 if ( $json_type eq 'number' && $decimals_to_pattern ) {
338 4 100 66     12 if ( $column_info->{size} && ref $column_info->{size} eq 'ARRAY' ) {
339 1         3 $json_schema{properties}->{ $column }->{type} = 'string';
340 1         14 $json_schema{properties}->{ $column }->{pattern} = $self->_get_decimal_pattern( $column_info->{size} );
341             }
342             }
343              
344             # JSON schema field patterns
345 195 100       2670 if ( $self->pattern_map->{ $column_info->{data_type} } ) {
346 30         518 $json_schema{properties}->{ $column }->{pattern} = $self->pattern_map->{ $column_info->{data_type} };
347             }
348              
349             # JSON schema property description
350 195 50 33     1479 if ( ! $json_schema{properties}->{ $column }->{description} && $has_schema_property_description ) {
351             my $property_description = $self->_get_json_schema_property_description(
352             $overwrite_schema_property_keys->{ $column } // $column,
353 0   0     0 $json_schema{properties}->{ $column }
354             );
355 0         0 $json_schema{properties}->{ $column }->{description} = $property_description;
356             }
357              
358             # JSON schema custom additional properties
359 195 50       341 if ( $add_schema_properties ) {
360 0         0 foreach my $property_key ( keys %{ $add_schema_properties } ) {
  0         0  
361 0         0 $json_schema{properties}->{ $property_key } = $add_schema_properties->{ $property_key };
362             }
363             }
364              
365             # Overwrites: merge JSON schema property key values with custom ones
366 195 100       396 if ( my $overwrite_property = delete $overwrite_schema_properties->{ $column } ) {
367 2   50     6 my $action = delete $overwrite_property->{_action} // 'merge';
368              
369             $json_schema{properties}->{ $column } = {
370 2 100       9 %{ $action eq 'merge' ? $json_schema{properties}->{ $column } : {} },
371 2         3 %{ $overwrite_property }
  2         17  
372             };
373             }
374              
375             # Overwrite: replace JSON schema keys
376 195 100       432 if ( my $new_key = $overwrite_schema_property_keys->{ $column } ) {
377 2         6 $json_schema{properties}->{ $new_key } = delete $json_schema{properties}->{ $column };
378             }
379             }
380              
381             return {
382             %json_schema,
383 7         40 %{ $schema_overwrite },
  7         88  
384             };
385             }
386              
387             # Return DBIx result source column info for the given result class name
388             sub _get_column_info {
389 10     10   26543 my ( $self, $source ) = @_;
390              
391 10         106 return $self->schema->source($source)->columns_info;
392             }
393              
394             # Returns RegExp pattern for decimal numbers based on database field definition
395             sub _get_decimal_pattern {
396 1     1   4 my ( $self, $size ) = @_;
397              
398 1         2 my ( $x, $y ) = @{ $size };
  1         3  
399 1         8 return sprintf '^\d{1,%s}\.\d{0,%s}$', $x - $y, $y;
400             }
401              
402             # Generates somewhat logical field description based on type and length constraints
403             sub _get_json_schema_property_description {
404 0     0   0 my ( $self, $column, $property ) = @_;
405              
406 0 0       0 if ( ! $property->{type} ) {
407 0 0       0 if ( $property->{enum} ) {
408 0         0 return sprintf 'Enum list type, one of - %s', join( ', ', @{ $property->{enum} } );
  0         0  
409             }
410              
411 0         0 return '';
412             }
413              
414 0 0       0 return '' if $property->{type} eq 'object'; # no idea how to handle
415              
416 0         0 my %types;
417 0 0       0 if ( ref $property->{type} eq 'ARRAY' ) {
418 0         0 %types = map { $_ => 1 } @{ $property->{type} };
  0         0  
  0         0  
419             }
420             else {
421 0         0 $types{ $property->{type} } = 1;
422             }
423              
424 0         0 my $description = '';
425 0 0       0 $description .= 'Optional' if $types{null};
426              
427 0         0 my $type_part;
428 0 0       0 if ( grep { /^integer|number$/ } keys %types ) {
  0         0  
429 0         0 $type_part = 'numeric';
430             }
431             else {
432 0         0 ( $type_part ) = grep { $_ ne 'null' } keys %types; # lucky roll, last type that isn't 'null' should be legit
  0         0  
433             }
434              
435 0 0       0 $description .= $description ? " $type_part" : ucfirst $type_part;
436 0         0 $description .= sprintf ' type value for field %s', $column;
437              
438 0 0 0     0 if ( ( grep { /^integer$/ } keys %types ) && $property->{maximum} ) {
  0 0 0     0  
439 0   0     0 my $integer_example = $property->{default} // int rand $property->{maximum};
440 0         0 $description .= ' e.g. ' . $integer_example;
441             }
442 0         0 elsif ( ( grep { /^string$/ } keys %types ) && $property->{pattern} ) {
443 0         0 $description .= sprintf ' with pattern %s ', $property->{pattern};
444             }
445              
446 0         0 return $description;
447             }
448              
449             # Convert from DBIx field length to JSON schema field length based on field type
450             sub _set_json_schema_property_range {
451 139     139   251 my ( $self, $json_schema, $column_info, $column, $add_property_minimum_value ) = @_;
452              
453 139         1815 my $json_schema_min_type = $self->length_type_map->{ $self->type_map->{ $column_info->{data_type} } }->[0];
454 139         4382 my $json_schema_max_type = $self->length_type_map->{ $self->type_map->{ $column_info->{data_type} } }->[1];
455              
456 139         2556 my $json_schema_min = $self->_get_json_schema_property_min_max_value( $column_info, 0 );
457 139         4652 my $json_schema_max = $self->_get_json_schema_property_min_max_value( $column_info, 1 );
458              
459             # bump min value to 1 (don't see how this starts from negative)
460 139 50       4549 $json_schema_min = 1 if $column_info->{is_auto_increment};
461              
462 139 100       244 $json_schema->{properties}->{ $column }->{ $json_schema_min_type } = $json_schema_min
463             if $add_property_minimum_value;
464 139         241 $json_schema->{properties}->{ $column }->{ $json_schema_max_type } = $json_schema_max;
465              
466 139 100       231 if ( $column_info->{size} ) {
467 34         65 $json_schema->{properties}->{ $column }->{ $json_schema_max_type } = $column_info->{size};
468             }
469              
470 139         220 return;
471             }
472              
473             # Returns min/max value from DBIx result field definition or lookup from defaults
474             sub _get_json_schema_property_min_max_value {
475 278     278   424 my ( $self, $column_info, $range ) = @_;
476              
477 278 0 33     496 if ( $column_info->{extra} && $column_info->{extra}->{unsigned} ) { # no autovivification
478 0         0 return $self->length_map->{ $column_info->{data_type} }->{unsigned}->[ $range ];
479             }
480              
481             return ref $self->length_map->{ $column_info->{data_type} } eq 'ARRAY' ? $self->length_map->{ $column_info->{data_type} }->[ $range ]
482 278 100       3375 : $self->length_map->{ $column_info->{data_type} }->{signed}->[ $range ];
483             }
484              
485             =head1 SEE ALSO
486              
487             L - Result source object
488              
489             =head1 AUTHOR
490              
491             malishew - C
492              
493             =head1 LICENSE
494              
495             This library is free software; you can redistribute it and/or modify it under
496             the same terms as Perl itself. If you would like to contribute documentation
497             or file a bug report then please raise an issue / pull request:
498              
499             https://github.com/Humanstate/p5-dbix-result-convert-jsonschema
500              
501             =cut
502              
503             __PACKAGE__->meta->make_immutable;