/**
* @license
* Copyright The Closure Library Authors.
* SPDX-License-Identifier: Apache-2.0
*/
/**
* @fileoverview Test helpers to compare goog.proto2.Messages.
*/
goog.setTestOnly('goog.testing.proto2');
goog.provide('goog.testing.proto2');
goog.require('goog.proto2.Message');
goog.require('goog.proto2.ObjectSerializer');
goog.require('goog.testing.asserts');
/**
* Compares two goog.proto2.Message instances of the same type.
* @param {!goog.proto2.Message} expected First message.
* @param {!goog.proto2.Message} actual Second message.
* @param {string} path Path to the messages.
* @return {string} A string describing where they differ. Empty string if they
* are equal.
* @private
*/
goog.testing.proto2.findDifferences_ = function(expected, actual, path) {
'use strict';
const fields = expected.getDescriptor().getFields();
for (let i = 0; i < fields.length; i++) {
const field = fields[i];
const newPath = (path ? path + '/' : '') + field.getName();
if (expected.has(field) && !actual.has(field)) {
return newPath + ' should be present';
}
if (!expected.has(field) && actual.has(field)) {
return newPath + ' should not be present';
}
if (expected.has(field)) {
const isComposite = field.isCompositeType();
if (field.isRepeated()) {
const expectedCount = expected.countOf(field);
const actualCount = actual.countOf(field);
if (expectedCount != actualCount) {
return newPath + ' should have ' + expectedCount + ' items, ' +
'but has ' + actualCount;
}
for (let j = 0; j < expectedCount; j++) {
const expectedItem = expected.get(field, j);
const actualItem = actual.get(field, j);
if (isComposite) {
const itemDiff = goog.testing.proto2.findDifferences_(
/** @type {!goog.proto2.Message} */ (expectedItem),
/** @type {!goog.proto2.Message} */ (actualItem),
newPath + '[' + j + ']');
if (itemDiff) {
return itemDiff;
}
} else {
if (expectedItem != actualItem) {
return newPath + '[' + j + '] should be ' + expectedItem +
', but was ' + actualItem;
}
}
}
} else {
const expectedValue = expected.get(field);
const actualValue = actual.get(field);
if (isComposite) {
const diff = goog.testing.proto2.findDifferences_(
/** @type {!goog.proto2.Message} */ (expectedValue),
/** @type {!goog.proto2.Message} */ (actualValue), newPath);
if (diff) {
return diff;
}
} else {
if (expectedValue != actualValue) {
return newPath + ' should be ' + expectedValue + ', but was ' +
actualValue;
}
}
}
}
}
return '';
};
/**
* Compares two goog.proto2.Message objects. Gives more readable output than
* assertObjectEquals on mismatch.
* @param {!goog.proto2.Message} expected Expected proto2 message.
* @param {!goog.proto2.Message} actual Actual proto2 message.
* @param {string=} opt_failureMessage Failure message when the values don't
* match.
*/
goog.testing.proto2.assertEquals = function(
expected, actual, opt_failureMessage) {
'use strict';
const failureSummary = opt_failureMessage || '';
if (!(expected instanceof goog.proto2.Message) ||
!(actual instanceof goog.proto2.Message)) {
goog.testing.asserts.raiseException(
failureSummary,
'Bad arguments were passed to goog.testing.proto2.assertEquals');
}
if (expected.constructor != actual.constructor) {
goog.testing.asserts.raiseException(
failureSummary, 'Message type mismatch: ' +
expected.getDescriptor().getFullName() + ' != ' +
actual.getDescriptor().getFullName());
}
const diff = goog.testing.proto2.findDifferences_(expected, actual, '');
if (diff) {
goog.testing.asserts.raiseException(failureSummary, diff);
}
};
/**
* Helper function to quickly build protocol buffer messages from JSON objects.
* @param {function(new:MessageType)} messageCtor A constructor that
* creates a `goog.proto2.Message` subclass instance.
* @param {!Object} json JSON object which uses field names as keys.
* @return {MessageType} The deserialized protocol buffer.
* @template MessageType
*/
goog.testing.proto2.fromObject = function(messageCtor, json) {
'use strict';
const serializer = new goog.proto2.ObjectSerializer(
goog.proto2.ObjectSerializer.KeyOption.NAME);
const message = new messageCtor;
serializer.deserializeTo(message, json);
return message;
};