| // Protocol Buffers - Google's data interchange format |
| // Copyright 2008 Google Inc. All rights reserved. |
| // https://developers.google.com/protocol-buffers/ |
| // |
| // Redistribution and use in source and binary forms, with or without |
| // modification, are permitted provided that the following conditions are |
| // met: |
| // |
| // * Redistributions of source code must retain the above copyright |
| // notice, this list of conditions and the following disclaimer. |
| // * Redistributions in binary form must reproduce the above |
| // copyright notice, this list of conditions and the following disclaimer |
| // in the documentation and/or other materials provided with the |
| // distribution. |
| // * Neither the name of Google Inc. nor the names of its |
| // contributors may be used to endorse or promote products derived from |
| // this software without specific prior written permission. |
| // |
| // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS |
| // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT |
| // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR |
| // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT |
| // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, |
| // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT |
| // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, |
| // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY |
| // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
| // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE |
| // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
| |
| #import "GPBTestUtilities.h" |
| |
| #import <objc/runtime.h> |
| |
| #import "GPBDescriptor_PackagePrivate.h" |
| #import "google/protobuf/Unittest.pbobjc.h" |
| #import "google/protobuf/UnittestObjc.pbobjc.h" |
| #import "google/protobuf/Descriptor.pbobjc.h" |
| |
| @interface DescriptorTests : GPBTestCase |
| @end |
| |
| @implementation DescriptorTests |
| |
| - (void)testDescriptor_containingType { |
| GPBDescriptor *testAllTypesDesc = [TestAllTypes descriptor]; |
| GPBDescriptor *nestedMessageDesc = [TestAllTypes_NestedMessage descriptor]; |
| XCTAssertNil(testAllTypesDesc.containingType); |
| XCTAssertNotNil(nestedMessageDesc.containingType); |
| XCTAssertEqual(nestedMessageDesc.containingType, testAllTypesDesc); // Ptr comparison |
| } |
| |
| - (void)testDescriptor_fullName { |
| GPBDescriptor *testAllTypesDesc = [TestAllTypes descriptor]; |
| XCTAssertEqualObjects(testAllTypesDesc.fullName, @"protobuf_unittest.TestAllTypes"); |
| GPBDescriptor *nestedMessageDesc = [TestAllTypes_NestedMessage descriptor]; |
| XCTAssertEqualObjects(nestedMessageDesc.fullName, @"protobuf_unittest.TestAllTypes.NestedMessage"); |
| |
| // Prefixes removed. |
| GPBDescriptor *descDesc = [GPBDescriptorProto descriptor]; |
| XCTAssertEqualObjects(descDesc.fullName, @"google.protobuf.DescriptorProto"); |
| GPBDescriptor *descExtRngDesc = [GPBDescriptorProto_ExtensionRange descriptor]; |
| XCTAssertEqualObjects(descExtRngDesc.fullName, @"google.protobuf.DescriptorProto.ExtensionRange"); |
| |
| // Things that get "_Class" added. |
| GPBDescriptor *pointDesc = [Point_Class descriptor]; |
| XCTAssertEqualObjects(pointDesc.fullName, @"protobuf_unittest.Point"); |
| GPBDescriptor *pointRectDesc = [Point_Rect descriptor]; |
| XCTAssertEqualObjects(pointRectDesc.fullName, @"protobuf_unittest.Point.Rect"); |
| } |
| |
| - (void)testFieldDescriptor { |
| GPBDescriptor *descriptor = [TestAllTypes descriptor]; |
| |
| // Nested Enum |
| GPBFieldDescriptor *fieldDescriptorWithName = |
| [descriptor fieldWithName:@"optionalNestedEnum"]; |
| XCTAssertNotNil(fieldDescriptorWithName); |
| GPBFieldDescriptor *fieldDescriptorWithNumber = |
| [descriptor fieldWithNumber:21]; |
| XCTAssertNotNil(fieldDescriptorWithNumber); |
| XCTAssertEqual(fieldDescriptorWithName, fieldDescriptorWithNumber); |
| XCTAssertNotNil(fieldDescriptorWithNumber.enumDescriptor); |
| XCTAssertEqualObjects(fieldDescriptorWithNumber.enumDescriptor.name, |
| @"TestAllTypes_NestedEnum"); |
| XCTAssertEqual(fieldDescriptorWithName.number, fieldDescriptorWithNumber.number); |
| XCTAssertEqual(fieldDescriptorWithName.dataType, GPBDataTypeEnum); |
| |
| // Foreign Enum |
| fieldDescriptorWithName = [descriptor fieldWithName:@"optionalForeignEnum"]; |
| XCTAssertNotNil(fieldDescriptorWithName); |
| fieldDescriptorWithNumber = [descriptor fieldWithNumber:22]; |
| XCTAssertNotNil(fieldDescriptorWithNumber); |
| XCTAssertEqual(fieldDescriptorWithName, fieldDescriptorWithNumber); |
| XCTAssertNotNil(fieldDescriptorWithNumber.enumDescriptor); |
| XCTAssertEqualObjects(fieldDescriptorWithNumber.enumDescriptor.name, |
| @"ForeignEnum"); |
| XCTAssertEqual(fieldDescriptorWithName.number, fieldDescriptorWithNumber.number); |
| XCTAssertEqual(fieldDescriptorWithName.dataType, GPBDataTypeEnum); |
| |
| // Import Enum |
| fieldDescriptorWithName = [descriptor fieldWithName:@"optionalImportEnum"]; |
| XCTAssertNotNil(fieldDescriptorWithName); |
| fieldDescriptorWithNumber = [descriptor fieldWithNumber:23]; |
| XCTAssertNotNil(fieldDescriptorWithNumber); |
| XCTAssertEqual(fieldDescriptorWithName, fieldDescriptorWithNumber); |
| XCTAssertNotNil(fieldDescriptorWithNumber.enumDescriptor); |
| XCTAssertEqualObjects(fieldDescriptorWithNumber.enumDescriptor.name, |
| @"ImportEnum"); |
| XCTAssertEqual(fieldDescriptorWithName.number, fieldDescriptorWithNumber.number); |
| XCTAssertEqual(fieldDescriptorWithName.dataType, GPBDataTypeEnum); |
| |
| // Nested Message |
| fieldDescriptorWithName = [descriptor fieldWithName:@"optionalNestedMessage"]; |
| XCTAssertNotNil(fieldDescriptorWithName); |
| fieldDescriptorWithNumber = [descriptor fieldWithNumber:18]; |
| XCTAssertNotNil(fieldDescriptorWithNumber); |
| XCTAssertEqual(fieldDescriptorWithName, fieldDescriptorWithNumber); |
| XCTAssertNil(fieldDescriptorWithNumber.enumDescriptor); |
| XCTAssertEqual(fieldDescriptorWithName.number, fieldDescriptorWithNumber.number); |
| XCTAssertEqual(fieldDescriptorWithName.dataType, GPBDataTypeMessage); |
| |
| // Foreign Message |
| fieldDescriptorWithName = |
| [descriptor fieldWithName:@"optionalForeignMessage"]; |
| XCTAssertNotNil(fieldDescriptorWithName); |
| fieldDescriptorWithNumber = [descriptor fieldWithNumber:19]; |
| XCTAssertNotNil(fieldDescriptorWithNumber); |
| XCTAssertEqual(fieldDescriptorWithName, fieldDescriptorWithNumber); |
| XCTAssertNil(fieldDescriptorWithNumber.enumDescriptor); |
| XCTAssertEqual(fieldDescriptorWithName.number, fieldDescriptorWithNumber.number); |
| XCTAssertEqual(fieldDescriptorWithName.dataType, GPBDataTypeMessage); |
| |
| // Import Message |
| fieldDescriptorWithName = [descriptor fieldWithName:@"optionalImportMessage"]; |
| XCTAssertNotNil(fieldDescriptorWithName); |
| fieldDescriptorWithNumber = [descriptor fieldWithNumber:20]; |
| XCTAssertNotNil(fieldDescriptorWithNumber); |
| XCTAssertEqual(fieldDescriptorWithName, fieldDescriptorWithNumber); |
| XCTAssertNil(fieldDescriptorWithNumber.enumDescriptor); |
| XCTAssertEqual(fieldDescriptorWithName.number, fieldDescriptorWithNumber.number); |
| XCTAssertEqual(fieldDescriptorWithName.dataType, GPBDataTypeMessage); |
| |
| // Some failed lookups. |
| XCTAssertNil([descriptor fieldWithName:@"NOT THERE"]); |
| XCTAssertNil([descriptor fieldWithNumber:9876543]); |
| } |
| |
| - (void)testEnumDescriptor { |
| GPBEnumDescriptor *descriptor = TestAllTypes_NestedEnum_EnumDescriptor(); |
| |
| NSString *enumName = [descriptor enumNameForValue:1]; |
| XCTAssertNotNil(enumName); |
| int32_t value; |
| XCTAssertTrue( |
| [descriptor getValue:&value forEnumName:@"TestAllTypes_NestedEnum_Foo"]); |
| XCTAssertTrue( |
| [descriptor getValue:NULL forEnumName:@"TestAllTypes_NestedEnum_Foo"]); |
| XCTAssertEqual(value, TestAllTypes_NestedEnum_Foo); |
| |
| enumName = [descriptor enumNameForValue:2]; |
| XCTAssertNotNil(enumName); |
| XCTAssertTrue( |
| [descriptor getValue:&value forEnumName:@"TestAllTypes_NestedEnum_Bar"]); |
| XCTAssertEqual(value, TestAllTypes_NestedEnum_Bar); |
| |
| enumName = [descriptor enumNameForValue:3]; |
| XCTAssertNotNil(enumName); |
| XCTAssertTrue( |
| [descriptor getValue:&value forEnumName:@"TestAllTypes_NestedEnum_Baz"]); |
| XCTAssertEqual(value, TestAllTypes_NestedEnum_Baz); |
| |
| // TextFormat |
| enumName = [descriptor textFormatNameForValue:1]; |
| XCTAssertNotNil(enumName); |
| XCTAssertTrue([descriptor getValue:&value forEnumTextFormatName:@"FOO"]); |
| XCTAssertEqual(value, TestAllTypes_NestedEnum_Foo); |
| XCTAssertNil([descriptor textFormatNameForValue:99999]); |
| |
| // Bad values |
| enumName = [descriptor enumNameForValue:0]; |
| XCTAssertNil(enumName); |
| XCTAssertFalse([descriptor getValue:&value forEnumName:@"Unknown"]); |
| XCTAssertFalse([descriptor getValue:NULL forEnumName:@"Unknown"]); |
| XCTAssertFalse([descriptor getValue:&value |
| forEnumName:@"TestAllTypes_NestedEnum_Unknown"]); |
| XCTAssertFalse([descriptor getValue:NULL |
| forEnumName:@"TestAllTypes_NestedEnum_Unknown"]); |
| XCTAssertFalse([descriptor getValue:NULL forEnumTextFormatName:@"Unknown"]); |
| XCTAssertFalse([descriptor getValue:&value forEnumTextFormatName:@"Unknown"]); |
| } |
| |
| - (void)testEnumDescriptorIntrospection { |
| GPBEnumDescriptor *descriptor = TestAllTypes_NestedEnum_EnumDescriptor(); |
| |
| XCTAssertEqual(descriptor.enumNameCount, 4U); |
| XCTAssertEqualObjects([descriptor getEnumNameForIndex:0], |
| @"TestAllTypes_NestedEnum_Foo"); |
| XCTAssertEqualObjects([descriptor getEnumTextFormatNameForIndex:0], @"FOO"); |
| XCTAssertEqualObjects([descriptor getEnumNameForIndex:1], |
| @"TestAllTypes_NestedEnum_Bar"); |
| XCTAssertEqualObjects([descriptor getEnumTextFormatNameForIndex:1], @"BAR"); |
| XCTAssertEqualObjects([descriptor getEnumNameForIndex:2], |
| @"TestAllTypes_NestedEnum_Baz"); |
| XCTAssertEqualObjects([descriptor getEnumTextFormatNameForIndex:2], @"BAZ"); |
| XCTAssertEqualObjects([descriptor getEnumNameForIndex:3], |
| @"TestAllTypes_NestedEnum_Neg"); |
| XCTAssertEqualObjects([descriptor getEnumTextFormatNameForIndex:3], @"NEG"); |
| } |
| |
| - (void)testEnumDescriptorIntrospectionWithAlias { |
| GPBEnumDescriptor *descriptor = TestEnumWithDupValue_EnumDescriptor(); |
| NSString *enumName; |
| int32_t value; |
| |
| XCTAssertEqual(descriptor.enumNameCount, 5U); |
| |
| enumName = [descriptor getEnumNameForIndex:0]; |
| XCTAssertEqualObjects(enumName, @"TestEnumWithDupValue_Foo1"); |
| XCTAssertTrue([descriptor getValue:&value forEnumName:enumName]); |
| XCTAssertEqual(value, 1); |
| XCTAssertEqualObjects([descriptor getEnumTextFormatNameForIndex:0], @"FOO1"); |
| |
| enumName = [descriptor getEnumNameForIndex:1]; |
| XCTAssertEqualObjects(enumName, @"TestEnumWithDupValue_Bar1"); |
| XCTAssertTrue([descriptor getValue:&value forEnumName:enumName]); |
| XCTAssertEqual(value, 2); |
| XCTAssertEqualObjects([descriptor getEnumTextFormatNameForIndex:1], @"BAR1"); |
| |
| enumName = [descriptor getEnumNameForIndex:2]; |
| XCTAssertEqualObjects(enumName, @"TestEnumWithDupValue_Baz"); |
| XCTAssertTrue([descriptor getValue:&value forEnumName:enumName]); |
| XCTAssertEqual(value, 3); |
| XCTAssertEqualObjects([descriptor getEnumTextFormatNameForIndex:2], @"BAZ"); |
| |
| enumName = [descriptor getEnumNameForIndex:3]; |
| XCTAssertEqualObjects(enumName, @"TestEnumWithDupValue_Foo2"); |
| XCTAssertTrue([descriptor getValue:&value forEnumName:enumName]); |
| XCTAssertEqual(value, 1); |
| XCTAssertEqualObjects([descriptor getEnumTextFormatNameForIndex:3], @"FOO2"); |
| |
| enumName = [descriptor getEnumNameForIndex:4]; |
| XCTAssertEqualObjects(enumName, @"TestEnumWithDupValue_Bar2"); |
| XCTAssertTrue([descriptor getValue:&value forEnumName:enumName]); |
| XCTAssertEqual(value, 2); |
| XCTAssertEqualObjects([descriptor getEnumTextFormatNameForIndex:4], @"BAR2"); |
| |
| } |
| |
| - (void)testEnumValueValidator { |
| GPBDescriptor *descriptor = [TestAllTypes descriptor]; |
| GPBFieldDescriptor *fieldDescriptor = |
| [descriptor fieldWithName:@"optionalNestedEnum"]; |
| |
| // Valid values |
| XCTAssertTrue([fieldDescriptor isValidEnumValue:1]); |
| XCTAssertTrue([fieldDescriptor isValidEnumValue:2]); |
| XCTAssertTrue([fieldDescriptor isValidEnumValue:3]); |
| XCTAssertTrue([fieldDescriptor isValidEnumValue:-1]); |
| |
| // Invalid values |
| XCTAssertFalse([fieldDescriptor isValidEnumValue:4]); |
| XCTAssertFalse([fieldDescriptor isValidEnumValue:0]); |
| XCTAssertFalse([fieldDescriptor isValidEnumValue:-2]); |
| } |
| |
| - (void)testOneofDescriptor { |
| GPBDescriptor *descriptor = [TestOneof2 descriptor]; |
| |
| // All fields should be listed. |
| XCTAssertEqual(descriptor.fields.count, 17U); |
| |
| // There are two oneofs in there. |
| XCTAssertEqual(descriptor.oneofs.count, 2U); |
| |
| GPBFieldDescriptor *fooStringField = |
| [descriptor fieldWithNumber:TestOneof2_FieldNumber_FooString]; |
| XCTAssertNotNil(fooStringField); |
| GPBFieldDescriptor *barStringField = |
| [descriptor fieldWithNumber:TestOneof2_FieldNumber_BarString]; |
| XCTAssertNotNil(barStringField); |
| |
| // Check the oneofs to have what is expected. |
| |
| GPBOneofDescriptor *oneofFoo = [descriptor oneofWithName:@"foo"]; |
| XCTAssertNotNil(oneofFoo); |
| XCTAssertEqual(oneofFoo.fields.count, 9U); |
| |
| // Pointer comparisons. |
| XCTAssertEqual([oneofFoo fieldWithNumber:TestOneof2_FieldNumber_FooString], |
| fooStringField); |
| XCTAssertEqual([oneofFoo fieldWithName:@"fooString"], fooStringField); |
| |
| GPBOneofDescriptor *oneofBar = [descriptor oneofWithName:@"bar"]; |
| XCTAssertNotNil(oneofBar); |
| XCTAssertEqual(oneofBar.fields.count, 6U); |
| |
| // Pointer comparisons. |
| XCTAssertEqual([oneofBar fieldWithNumber:TestOneof2_FieldNumber_BarString], |
| barStringField); |
| XCTAssertEqual([oneofBar fieldWithName:@"barString"], barStringField); |
| |
| // Unknown oneof not found. |
| |
| XCTAssertNil([descriptor oneofWithName:@"mumble"]); |
| XCTAssertNil([descriptor oneofWithName:@"Foo"]); |
| |
| // Unknown oneof item. |
| |
| XCTAssertNil([oneofFoo fieldWithName:@"mumble"]); |
| XCTAssertNil([oneofFoo fieldWithNumber:666]); |
| |
| // Field exists, but not in this oneof. |
| |
| XCTAssertNil([oneofFoo fieldWithName:@"barString"]); |
| XCTAssertNil([oneofFoo fieldWithNumber:TestOneof2_FieldNumber_BarString]); |
| XCTAssertNil([oneofBar fieldWithName:@"fooString"]); |
| XCTAssertNil([oneofBar fieldWithNumber:TestOneof2_FieldNumber_FooString]); |
| |
| // Check pointers back to the enclosing oneofs. |
| // (pointer comparisions) |
| XCTAssertEqual(fooStringField.containingOneof, oneofFoo); |
| XCTAssertEqual(barStringField.containingOneof, oneofBar); |
| GPBFieldDescriptor *bazString = |
| [descriptor fieldWithNumber:TestOneof2_FieldNumber_BazString]; |
| XCTAssertNotNil(bazString); |
| XCTAssertNil(bazString.containingOneof); |
| } |
| |
| - (void)testExtensiondDescriptor { |
| Class msgClass = [TestAllExtensions class]; |
| Class packedMsgClass = [TestPackedExtensions class]; |
| |
| // Int |
| |
| GPBExtensionDescriptor *descriptor = [UnittestRoot optionalInt32Extension]; |
| XCTAssertNotNil(descriptor); |
| XCTAssertEqual(descriptor.containingMessageClass, msgClass); // ptr equality |
| XCTAssertFalse(descriptor.isPackable); |
| XCTAssertEqualObjects(descriptor.defaultValue, @0); |
| XCTAssertNil(descriptor.enumDescriptor); |
| |
| descriptor = [UnittestRoot defaultInt32Extension]; |
| XCTAssertNotNil(descriptor); |
| XCTAssertEqual(descriptor.containingMessageClass, msgClass); // ptr equality |
| XCTAssertFalse(descriptor.isPackable); |
| XCTAssertEqualObjects(descriptor.defaultValue, @41); |
| XCTAssertNil(descriptor.enumDescriptor); |
| |
| // Enum |
| |
| descriptor = [UnittestRoot optionalNestedEnumExtension]; |
| XCTAssertNotNil(descriptor); |
| XCTAssertEqual(descriptor.containingMessageClass, msgClass); // ptr equality |
| XCTAssertFalse(descriptor.isPackable); |
| XCTAssertEqual(descriptor.defaultValue, @1); |
| XCTAssertEqualObjects(descriptor.enumDescriptor.name, @"TestAllTypes_NestedEnum"); |
| |
| descriptor = [UnittestRoot defaultNestedEnumExtension]; |
| XCTAssertNotNil(descriptor); |
| XCTAssertEqual(descriptor.containingMessageClass, msgClass); // ptr equality |
| XCTAssertFalse(descriptor.isPackable); |
| XCTAssertEqual(descriptor.defaultValue, @2); |
| XCTAssertEqualObjects(descriptor.enumDescriptor.name, @"TestAllTypes_NestedEnum"); |
| |
| // Message |
| |
| descriptor = [UnittestRoot optionalNestedMessageExtension]; |
| XCTAssertNotNil(descriptor); |
| XCTAssertEqual(descriptor.containingMessageClass, msgClass); // ptr equality |
| XCTAssertFalse(descriptor.isPackable); |
| XCTAssertNil(descriptor.defaultValue); |
| XCTAssertNil(descriptor.enumDescriptor); |
| |
| // Repeated Int |
| |
| descriptor = [UnittestRoot repeatedInt32Extension]; |
| XCTAssertNotNil(descriptor); |
| XCTAssertEqual(descriptor.containingMessageClass, msgClass); // ptr equality |
| XCTAssertFalse(descriptor.isPackable); |
| XCTAssertNil(descriptor.defaultValue); |
| XCTAssertNil(descriptor.enumDescriptor); |
| |
| descriptor = [UnittestRoot packedInt32Extension]; |
| XCTAssertNotNil(descriptor); |
| XCTAssertEqual(descriptor.containingMessageClass, packedMsgClass); // ptr equality |
| XCTAssertTrue(descriptor.isPackable); |
| XCTAssertNil(descriptor.defaultValue); |
| XCTAssertNil(descriptor.enumDescriptor); |
| |
| // Repeated Enum |
| |
| descriptor = [UnittestRoot repeatedNestedEnumExtension]; |
| XCTAssertNotNil(descriptor); |
| XCTAssertEqual(descriptor.containingMessageClass, msgClass); // ptr equality |
| XCTAssertFalse(descriptor.isPackable); |
| XCTAssertNil(descriptor.defaultValue); |
| XCTAssertEqualObjects(descriptor.enumDescriptor.name, @"TestAllTypes_NestedEnum"); |
| |
| descriptor = [UnittestRoot packedEnumExtension]; |
| XCTAssertNotNil(descriptor); |
| XCTAssertEqual(descriptor.containingMessageClass, packedMsgClass); // ptr equality |
| XCTAssertTrue(descriptor.isPackable); |
| XCTAssertNil(descriptor.defaultValue); |
| XCTAssertEqualObjects(descriptor.enumDescriptor.name, @"ForeignEnum"); |
| |
| // Repeated Message |
| |
| descriptor = [UnittestRoot repeatedNestedMessageExtension]; |
| XCTAssertNotNil(descriptor); |
| XCTAssertEqual(descriptor.containingMessageClass, msgClass); // ptr equality |
| XCTAssertFalse(descriptor.isPackable); |
| XCTAssertNil(descriptor.defaultValue); |
| XCTAssertNil(descriptor.enumDescriptor); |
| |
| // Compare (used internally for serialization). |
| |
| GPBExtensionDescriptor *ext1 = [UnittestRoot optionalInt32Extension]; |
| XCTAssertEqual(ext1.fieldNumber, 1u); |
| GPBExtensionDescriptor *ext2 = [UnittestRoot optionalInt64Extension]; |
| XCTAssertEqual(ext2.fieldNumber, 2u); |
| |
| XCTAssertEqual([ext1 compareByFieldNumber:ext2], NSOrderedAscending); |
| XCTAssertEqual([ext2 compareByFieldNumber:ext1], NSOrderedDescending); |
| XCTAssertEqual([ext1 compareByFieldNumber:ext1], NSOrderedSame); |
| } |
| |
| @end |