From ee15e3377d897fdaec61bafbdf2bf5d3557391e0 Mon Sep 17 00:00:00 2001 From: codingconcepts Date: Tue, 14 Aug 2018 19:59:58 +0100 Subject: [PATCH] Fixed issue whereby empty RawJSON objects would cause MarshalJSON to fail --- main-packr.go | 4 ++-- tmpl/x_helpers.html | 3 +++ tmpl/x_helpers_test.html | 27 +++++++++++++++++++++++++++ 3 files changed, 32 insertions(+), 2 deletions(-) diff --git a/main-packr.go b/main-packr.go index 236101d..4ccd76c 100644 --- a/main-packr.go +++ b/main-packr.go @@ -12,6 +12,6 @@ func init() { packr.PackJSONBytes("./tmpl", "model.html", "\"e3tkZWZpbmUgIm1vZGVsIn19CnBhY2thZ2Uge3sgLlBhY2thZ2VOYW1lIH19CgovKi0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLSsKfCBDb2RlIGdlbmVyYXRlZCBieSBtb2RlbGdlbiB8CnwgICAgICAgIERPIE5PVCBFRElULiAgICAgICAgfAorLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tKi8KCi8vIElmIHlvdSB3YW50IHRvIGV4dGVuZCBiZWhhdmlvdXI6Ci8vIFBsZWFzZSBjcmVhdGUgYSBjdXN0b20gZmlsZSBpbiB0aGlzIGRpcmVjdG9yeQovLyBhbmQgY3JlYXRlIHRoZSBtZXRob2RzIHRoZXJlIHRvIGF2b2lkIG92ZXJ3cml0aW5nIHlvdXIgY29kZSB3aXRoIHRoZSBnZW5lcmF0ZWQgb25lLgoKaW1wb3J0ICgKImZtdCIKICAgIHt7IHJhbmdlICRrLCAkdjo9IC5Nb2RlbC5JbXBvcnRzIH19CiAgICAie3ska319IgogICAge3stIGVuZCB9fQopCgovLyB7ey5Nb2RlbC5OYW1lfX0gcmVwcmVzZW50cyBhIHJvdyBpbiB0aGUge3suTW9kZWwuVGFibGVOYW1lfX0gdGFibGUKdHlwZSB7ey5Nb2RlbC5OYW1lfX0gc3RydWN0IHsKICAgIHt7IHJhbmdlICRrLCAkdjo9IC5Nb2RlbC5GaWVsZHMgfX0KICAgICAgICB7eyAkdi5OYW1lIH19IHt7ICR2LlR5cGUgfX0gYGpzb246Int7JHYuQ29sdW1uTmFtZX19ImAKICAgIHt7LSBlbmQgfX0Kb2Zmc2V0IGludApsaW1pdCBpbnQKfQoKLy8gSW5zZXJ0IGEgbmV3IHt7Lk1vZGVsLk5hbWV9fSByb3cgaW4gdGhlIHt7Lk1vZGVsLlRhYmxlTmFtZX19IHRhYmxlCmZ1bmMgKHt7LlJlY2VpdmVyfX0gKnt7Lk1vZGVsLk5hbWV9fSkgSW5zZXJ0KHF1IFF1ZXJ5ZXIpIChsYXN0SW5zZXJ0SUQgaW50NjQsIGVyciBlcnJvcikgewogICAgY29uc3Qgc3RtdCA9ICJJTlNFUlQgSU5UTyB7ey5Nb2RlbC5UYWJsZU5hbWV9fSAoe3suTW9kZWwuRmllbGRzIHwgaW5zZXJ0X2ZpZWxkc319KSBWQUxVRVMgKHt7Lk1vZGVsLkZpZWxkcyB8IGluc2VydF92YWx1ZXN9fSkiCiAgICByZXMsIGVyciA6PSBxdS5FeGVjKHN0bXQsIHt7IC4gfCBpbnNlcnRfYXJncyB9fSkKICAgIGlmIGVyciAhPSBuaWwgewogICAgICAgIHJldHVybiAwLCBlcnIKICAgIH0KICAgIHJldHVybiByZXMuTGFzdEluc2VydElkKCkKfQoKLy8gVXBkYXRlIGFuIGV4aXN0aW5nIHt7Lk1vZGVsLk5hbWV9fSByb3cgaW4gdGhlIHt7Lk1vZGVsLlRhYmxlTmFtZX19IHRhYmxlLgpmdW5jICh7ey5SZWNlaXZlcn19ICp7ey5Nb2RlbC5OYW1lfX0pIFVwZGF0ZShxdSBRdWVyeWVyLCBpZCBpbnQ2NCkgKGludDY0LCBlcnJvcikgewogICAgY29uc3Qgc3RtdCA9ICJVUERBVEUge3suTW9kZWwuVGFibGVOYW1lfX0gU0VUIHt7IC4gfCB1cGRhdGVfdmFsdWVzIH19IFdIRVJFIGlkID0gPyIKICAgIHJlc3VsdCwgZXJyIDo9IHF1LkV4ZWMoc3RtdCwge3sgLiB8IHVwZGF0ZV9hcmdzIH19ICwgaWQpCiAgICBpZiBlcnIgIT0gbmlsIHsKICAgICAgICByZXR1cm4gMCwgZXJyCiAgICB9CiAgICByZXR1cm4gcmVzdWx0LlJvd3NBZmZlY3RlZCgpCn0KCi8vIFVwc2VydCBpbnNlcnRzIGEgbmV3IHt7Lk1vZGVsLk5hbWV9fSByb3cgaW4gdGhlIHt7Lk1vZGVsLlRhYmxlTmFtZX19IHRhYmxlCi8vIGlmIHRoZSB1bmlxdWUgY29uc3RyYWludHMgYXJlIG5vdCBmb3VuZCwgb3RoZXJ3aXNlIGl0IHVwZGF0ZXMgaXQuCmZ1bmMgKHt7LlJlY2VpdmVyfX0gKnt7Lk1vZGVsLk5hbWV9fSkgVXBzZXJ0KHF1IFF1ZXJ5ZXIpIChsYXN0SW5zZXJ0SUQgaW50NjQsIGVyciBlcnJvcikgewogICAgY29uc3Qgc3RtdCA9ICJJTlNFUlQgSU5UTyB7ey5Nb2RlbC5UYWJsZU5hbWV9fSAoe3suTW9kZWwuRmllbGRzIHwgdXBzZXJ0X2ZpZWxkc319KSBWQUxVRVMgKHt7Lk1vZGVsLkZpZWxkcyB8IHVwc2VydF92YWx1ZXN9fSkgT04gRFVQTElDQVRFIEtFWSBVUERBVEUge3sgLiB8IHVwc2VydF9vbl9kdXBsaWNhdGUgfX0iCiAgICByZXMsIGVyciA6PSBxdS5FeGVjKHN0bXQsIHt7IC4gfCB1cHNlcnRfYXJncyB9fSkKICAgIGlmIGVyciAhPSBuaWwgewogICAgICAgIHJldHVybiAwLCBlcnIKICAgIH0KICAgIHJldHVybiByZXMuTGFzdEluc2VydElkKCkKfQoKLy8gRmluZCBhbiBleGlzdGluZyB7ey5Nb2RlbC5OYW1lfX0gcm93IGluIHRoZSB7ey5Nb2RlbC5UYWJsZU5hbWV9fSB0YWJsZQpmdW5jICh7ey5SZWNlaXZlcn19ICp7ey5Nb2RlbC5OYW1lfX0pIEZpbmQocXUgUXVlcnllciwgaWQgaW50NjQpIGVycm9yIHsKICAgIGNvbnN0IHN0bXQgPSAiU0VMRUNUICogRlJPTSB7ey5Nb2RlbC5UYWJsZU5hbWV9fSBXSEVSRSBpZCA9ID8iCiAgICByb3cgOj0gcXUuUXVlcnlSb3coc3RtdCwgaWQpCiAgICByZXR1cm4gcm93LlNjYW4oe3sgLiB8IHNjYW5fZmllbGRzfX0pCn0KCi8vIExvYWQgYWxsLCBvciBhIHN1YnNldCBvZiB7ey5Nb2RlbC5OYW1lfX0gcm93cyBmcm9tIHRoZSB7ey5Nb2RlbC5UYWJsZU5hbWV9fSB0YWJsZQpmdW5jICh7ey5SZWNlaXZlcn19ICp7ey5Nb2RlbC5OYW1lfX0pIExvYWQocXUgUXVlcnllcikgKHNldCBbXXt7Lk1vZGVsLk5hbWV9fSwgZXJyIGVycm9yKSB7CiAgICBzdG10IDo9ICJTRUxFQ1QgKiBGUk9NIHt7Lk1vZGVsLlRhYmxlTmFtZX19IgoKICAgIGlmIHt7LlJlY2VpdmVyfX0ubGltaXQgPT0gMCAmJiB7ey5SZWNlaXZlcn19Lm9mZnNldCA+IDAgewogICAgICAgIHJldHVybiBzZXQsIGZtdC5FcnJvcmYoImNhbm5vdCBxdWVyeSB3aXRoIG9mZnNldCBidXQgbm8gbGltaXQiKQogICAgfQoKICAgIGlmIHt7LlJlY2VpdmVyfX0ubGltaXQgPiAwIHsKICAgICAgICBzdG10ICs9IGZtdC5TcHJpbnRmKCIgTElNSVQgJWQiLCB7ey5SZWNlaXZlcn19LmxpbWl0KQogICAgfQogICAgaWYge3suUmVjZWl2ZXJ9fS5vZmZzZXQgPiAwIHsKICAgICAgICBzdG10ICs9IGZtdC5TcHJpbnRmKCIgT0ZGU0VUICVkIiwge3suUmVjZWl2ZXJ9fS5vZmZzZXQpCiAgICB9CiAgICBkZWZlciBmdW5jKCkgewogICAgICAgIHt7LlJlY2VpdmVyfX0ubGltaXQgPSAwCiAgICAgICAge3suUmVjZWl2ZXJ9fS5vZmZzZXQgPSAwCiAgICB9KCkKICAgIHJvd3MsIGVyciA6PSBxdS5RdWVyeShzdG10KQogICAgaWYgZXJyICE9IG5pbCB7CiAgICAgICAgcmV0dXJuCiAgICB9CiAgICBkZWZlciByb3dzLkNsb3NlKCkKICAgICAgICBmb3Igcm93cy5OZXh0KCkgewogICAgICAgICAgICB2YXIge3suUmVjZWl2ZXJ9fSB7ey5Nb2RlbC5OYW1lfX0KICAgICAgICAgICAgaWYgZXJyID0gcm93cy5TY2FuKHt7LiB8IHNjYW5fZmllbGRzfX0pOyBlcnIgIT0gbmlsIHsKICAgICAgICAgICAgcmV0dXJuCiAgICAgICAgfQogICAgICAgIHNldCA9IGFwcGVuZChzZXQsIHt7LlJlY2VpdmVyfX0pCiAgICB9CgogICAgcmV0dXJuCn0KCi8vIERlbGV0ZSBhbiBleGlzdGluZyB7ey5Nb2RlbC5OYW1lfX0gcm93IGZyb20gdGhlIHt7Lk1vZGVsLlRhYmxlTmFtZX19IHRhYmxlCmZ1bmMgKHt7LlJlY2VpdmVyfX0gKnt7Lk1vZGVsLk5hbWV9fSkgRGVsZXRlKHF1IFF1ZXJ5ZXIsIGlkIGludDY0KSAocm93c0FmZmVjdGVkIGludDY0LCBlcnIgZXJyb3IpIHsKICAgIGNvbnN0IHN0bXQgPSAiREVMRVRFIEZST00ge3suTW9kZWwuVGFibGVOYW1lfX0gV0hFUkUgaWQgPSA/IgogICAgcmVzdWx0LCBlcnIgOj0gcXUuRXhlYyhzdG10LCBpZCkKCWlmIGVyciAhPSBuaWwgewoJCXJldHVybgoJfQoKCXJldHVybiByZXN1bHQuUm93c0FmZmVjdGVkKCkKfQoKLy8gQ291bnQgdGhlIG51bWJlciBvZiByb3dzIGZyb20gdGhlIHt7Lk1vZGVsLlRhYmxlTmFtZX19IHRhYmxlCmZ1bmMoe3suUmVjZWl2ZXJ9fSAqe3suTW9kZWwuTmFtZX19KSBDb3VudChxdSBRdWVyeWVyKSAoY291bnQgaW50NjQsIGVyciBlcnJvcikgewogICAgY29uc3Qgc3RtdCA9ICJTRUxFQ1QgQ09VTlQoKikgRlJPTSB7ey5Nb2RlbC5UYWJsZU5hbWV9fSIKICAgIHJvdyA6PSBxdS5RdWVyeVJvdyhzdG10KQogICAgaWYgZXJyID0gcm93LlNjYW4oJmNvdW50KTsgZXJyICE9IG5pbCB7CiAgICAgICAgcmV0dXJuCiAgICB9CiAgICByZXR1cm4KfQoKLy8gRXhpc3RzIGNoZWNrcyBmb3IgdGhlIGl0ZW1zIGV4aXN0ZW5jZSBpbiB0aGUgZGF0YWJhc2UsIGJhc2VkIG9uIGl0J3MgaWQuCi8vIEFuIGVycm9yIHdpbGwgb25seSBiZSByZXR1cm5lZCBpZiBhIFNRTCByZWxhdGVkIGZhaWx1cmUgaGFwcGVucy4KLy8gSW4gYWxsIG90aGVyIGNhc2VzLCBhIGJvb2wgYW5kIG5pbCB3aWxsIHJldHVybi4KZnVuYyh7ey5SZWNlaXZlcn19ICp7ey5Nb2RlbC5OYW1lfX0pIEV4aXN0cyhxdSBRdWVyeWVyLCBpZCBpbnQ2NCkgKGV4aXN0cyBib29sLCBlcnIgZXJyb3IpIHsKICAgIGNvbnN0IHN0bXQgPSAiU0VMRUNUIEVYSVNUUyhTRUxFQ1QgMSBGUk9NIHt7Lk1vZGVsLlRhYmxlTmFtZX19IFdIRVJFIGlkID0gPyBMSU1JVCAxKSBBUyBgZXhpc3RzYCIKICAgIHZhciBjb3VudCBpbnQKICAgIHJvdyA6PSBxdS5RdWVyeVJvdyhzdG10LCBpZCkKICAgIGlmIGVyciA9IHJvdy5TY2FuKCZjb3VudCk7IGVyciAhPSBuaWwgewogICAgICAgIHJldHVybgogICAgfQogICAgcmV0dXJuIGNvdW50ID4gMCwgbmlsCn0KCi8vIFRhYmxlTmFtZSByZXR1cm5zIHRoZSB0YWJsZSBuYW1lCmZ1bmMgKHt7LlJlY2VpdmVyfX0gKnt7Lk1vZGVsLk5hbWV9fSkgVGFibGVOYW1lKCkgc3RyaW5nIHsKcmV0dXJuICJ7ey5Nb2RlbC5UYWJsZU5hbWV9fSIKfQoKLy8gU2V0TGltaXQgc2V0cyB0aGUgcXVlcnkgbGltaXQKZnVuYyAoe3suUmVjZWl2ZXJ9fSAqe3suTW9kZWwuTmFtZX19KSBTZXRMaW1pdChsaW1pdCBpbnQpICp7ey5Nb2RlbC5OYW1lfX0gewp7ey5SZWNlaXZlcn19LmxpbWl0ID0gbGltaXQKcmV0dXJuIHt7LlJlY2VpdmVyfX0KfQoKLy8gU2V0T2Zmc2V0IHNldHMgdGhlIHF1ZXJ5IG9mZnNldApmdW5jICh7ey5SZWNlaXZlcn19ICp7ey5Nb2RlbC5OYW1lfX0pIFNldE9mZnNldChvZmZzZXQgaW50KSAqe3suTW9kZWwuTmFtZX19IHsKe3suUmVjZWl2ZXJ9fS5vZmZzZXQgPSBvZmZzZXQKcmV0dXJuIHt7LlJlY2VpdmVyfX0KfQp7e2VuZH19Cgo=\"") packr.PackJSONBytes("./tmpl", "tmpl.go", "\"cGFja2FnZSB0bXBsCgppbXBvcnQgKAoJImZtdCIKCSJodG1sL3RlbXBsYXRlIgoJInN0cmluZ3MiCikKCnZhciBGdW5jTWFwID0gdGVtcGxhdGUuRnVuY01hcHsKCSJpbnNlcnRfZmllbGRzIjogR2V0SW5zZXJ0RmllbGRzLAoJImluc2VydF92YWx1ZXMiOiBHZXRJbnNlcnRWYWx1ZXMsCgkiaW5zZXJ0X2FyZ3MiOiAgIEdldEluc2VydEFyZ3MsCgkic2Nhbl9maWVsZHMiOiAgIEdldFNjYW5GaWVsZHMsCgkidXBkYXRlX2FyZ3MiOiAgIEdldFVwZGF0ZUFyZ3MsCgkidXBkYXRlX3ZhbHVlcyI6IEdldFVwZGF0ZVZhbHVlcywKCSJ1cHNlcnRfZmllbGRzIjogR2V0VXBzZXJ0RmllbGRzLAoJInVwc2VydF92YWx1ZXMiOiBHZXRVcHNlcnRWYWx1ZXMsCgkidXBzZXJ0X29uX2R1cGxpY2F0ZSI6IEdldFVwc2VydE9uRHVwbGljYXRlLAoJInVwc2VydF9hcmdzIjogR2V0VXBzZXJ0QXJncywKfQoKZnVuYyBHZXRJbnNlcnRGaWVsZHMoZmllbGRzIFtdVG1wbEZpZWxkKSBzdHJpbmcgewoJdmFyIHBhcnRzIFtdc3RyaW5nCglmb3IgXywgZmwgOj0gcmFuZ2UgZmllbGRzIHsKCQlpZiBmbC5Db2x1bW5OYW1lID09ICJpZCIgewoJCQljb250aW51ZQoJCX0KCQlwYXJ0cyA9IGFwcGVuZChwYXJ0cywgImAiK2ZsLkNvbHVtbk5hbWUrImAiKQoJfQoJcmV0dXJuIHN0cmluZ3MuSm9pbihwYXJ0cywgIiwgIikKfQoKZnVuYyBHZXRJbnNlcnRWYWx1ZXMoZmllbGRzIFtdVG1wbEZpZWxkKSBzdHJpbmcgewoJdmFyIHBhcnRzIFtdc3RyaW5nCglmb3IgXywgZmwgOj0gcmFuZ2UgZmllbGRzIHsKCQlzd2l0Y2ggZmwuQ29sdW1uTmFtZSB7CgkJY2FzZSAiaWQiOgoJCQljb250aW51ZQoJCWNhc2UgImNyZWF0ZWRfYXQiOgoJCQlwYXJ0cyA9IGFwcGVuZChwYXJ0cywgIk5PVygpIikKCQkJY29udGludWUKCQlkZWZhdWx0OgoJCQlwYXJ0cyA9IGFwcGVuZChwYXJ0cywgIj8iKQoJCX0KCX0KCXJldHVybiBzdHJpbmdzLkpvaW4ocGFydHMsICIsICIpCn0KCmZ1bmMgR2V0SW5zZXJ0QXJncyhtIFN0cnVjdFRtcGxEYXRhKSBzdHJpbmcgewoJdmFyIHBhcnRzIFtdc3RyaW5nCglmb3IgXywgZmwgOj0gcmFuZ2UgbS5Nb2RlbC5GaWVsZHMgewoJCXN3aXRjaCBmbC5OYW1lIHsKCQljYXNlICJJRCIsICJDcmVhdGVkQXQiOgoJCQljb250aW51ZQoJCX0KCQlwYXJ0cyA9IGFwcGVuZChwYXJ0cywgZm10LlNwcmludGYoIiVzLiVzIiwgbS5SZWNlaXZlciwgZmwuTmFtZSkpCgl9CglyZXR1cm4gc3RyaW5ncy5Kb2luKHBhcnRzLCAiLCAiKQp9CgpmdW5jIEdldFNjYW5GaWVsZHMobSBTdHJ1Y3RUbXBsRGF0YSkgdGVtcGxhdGUuSFRNTCB7Cgl2YXIgcGFydHMgW11zdHJpbmcKCWZvciBfLCBmbCA6PSByYW5nZSBtLk1vZGVsLkZpZWxkcyB7CgkJcGFydHMgPSBhcHBlbmQocGFydHMsIGZtdC5TcHJpbnRmKCImJXMuJXMiLCBtLlJlY2VpdmVyLCBmbC5OYW1lKSkKCX0KCXJldHVybiB0ZW1wbGF0ZS5IVE1MKHN0cmluZ3MuSm9pbihwYXJ0cywgIiwgIikpCn0KCmZ1bmMgR2V0VXBkYXRlQXJncyhtIFN0cnVjdFRtcGxEYXRhKSB0ZW1wbGF0ZS5IVE1MIHsKCXZhciBwYXJ0cyBbXXN0cmluZwoJZm9yIF8sIGZsIDo9IHJhbmdlIG0uTW9kZWwuRmllbGRzIHsKCQlzd2l0Y2ggZmwuTmFtZSB7CgkJY2FzZSAiSUQiLCAiQ3JlYXRlZEF0IiwgIlVwZGF0ZWRBdCI6CgkJCWNvbnRpbnVlCgkJfQoJCXBhcnRzID0gYXBwZW5kKHBhcnRzLCBmbXQuU3ByaW50ZigiJXMuJXMiLCBtLlJlY2VpdmVyLCBmbC5OYW1lKSkKCX0KCXJldHVybiB0ZW1wbGF0ZS5IVE1MKHN0cmluZ3MuSm9pbihwYXJ0cywgIiwgIikpCn0KCmZ1bmMgR2V0VXBkYXRlVmFsdWVzKG0gU3RydWN0VG1wbERhdGEpIHN0cmluZyB7Cgl2YXIgcGFydHMgW11zdHJpbmcKCWZvciBfLCBmbCA6PSByYW5nZSBtLk1vZGVsLkZpZWxkcyB7CgkJc3dpdGNoIGZsLk5hbWUgewoJCWNhc2UgIklEIiwgIkNyZWF0ZWRBdCI6CgkJCWNvbnRpbnVlCgkJY2FzZSAiVXBkYXRlZEF0IjoKCQkJcGFydHMgPSBhcHBlbmQocGFydHMsIGZtdC5TcHJpbnRmKCJgJXNgPVVUQ19USU1FU1RBTVAoKSIsIGZsLkNvbHVtbk5hbWUpKQoJCWRlZmF1bHQ6CgkJCXBhcnRzID0gYXBwZW5kKHBhcnRzLCBmbXQuU3ByaW50ZigiYCVzYD0/IiwgZmwuQ29sdW1uTmFtZSkpCgkJfQoJfQoJcmV0dXJuIHN0cmluZ3MuSm9pbihwYXJ0cywgIiwgIikKfQoKZnVuYyBHZXRVcHNlcnRGaWVsZHMoZmllbGRzIFtdVG1wbEZpZWxkKSBzdHJpbmcgewoJdmFyIHBhcnRzIFtdc3RyaW5nCglmb3IgXywgZmwgOj0gcmFuZ2UgZmllbGRzIHsKCQlwYXJ0cyA9IGFwcGVuZChwYXJ0cywgImAiK2ZsLkNvbHVtbk5hbWUrImAiKQoJfQoJcmV0dXJuIHN0cmluZ3MuSm9pbihwYXJ0cywgIiwgIikKfQoKZnVuYyBHZXRVcHNlcnRWYWx1ZXMoZmllbGRzIFtdVG1wbEZpZWxkKSBzdHJpbmcgewoJdmFyIHBhcnRzIFtdc3RyaW5nCglmb3IgXywgZmwgOj0gcmFuZ2UgZmllbGRzIHsKCQlzd2l0Y2ggZmwuQ29sdW1uTmFtZSB7CgkJY2FzZSAiY3JlYXRlZF9hdCI6CgkJCXBhcnRzID0gYXBwZW5kKHBhcnRzLCAiTk9XKCkiKQoJCQljb250aW51ZQoJCWRlZmF1bHQ6CgkJCXBhcnRzID0gYXBwZW5kKHBhcnRzLCAiPyIpCgkJfQoJfQoJcmV0dXJuIHN0cmluZ3MuSm9pbihwYXJ0cywgIiwgIikKfQoKZnVuYyBHZXRVcHNlcnRPbkR1cGxpY2F0ZShtIFN0cnVjdFRtcGxEYXRhKSBzdHJpbmcgewoJdmFyIHBhcnRzIFtdc3RyaW5nCglmb3IgXywgZmwgOj0gcmFuZ2UgbS5Nb2RlbC5GaWVsZHMgewoJCXN3aXRjaCBmbC5OYW1lIHsKCQljYXNlICJDcmVhdGVkQXQiOgoJCQljb250aW51ZQoJCWNhc2UgIklEIjoKCQkJcGFydHMgPSBhcHBlbmQocGFydHMsIGZtdC5TcHJpbnRmKCJgJXNgPUxBU1RfSU5TRVJUX0lEKGAlc2ApIiwgZmwuQ29sdW1uTmFtZSwgZmwuQ29sdW1uTmFtZSkpCgkJY2FzZSAiVXBkYXRlZEF0IjoKCQkJcGFydHMgPSBhcHBlbmQocGFydHMsIGZtdC5TcHJpbnRmKCJgJXNgPVVUQ19USU1FU1RBTVAoKSIsIGZsLkNvbHVtbk5hbWUpKQoJCWRlZmF1bHQ6CgkJCXBhcnRzID0gYXBwZW5kKHBhcnRzLCBmbXQuU3ByaW50ZigiYCVzYD1WQUxVRVMoYCVzYCkiLCBmbC5Db2x1bW5OYW1lLCBmbC5Db2x1bW5OYW1lKSkKCQl9Cgl9CglyZXR1cm4gc3RyaW5ncy5Kb2luKHBhcnRzLCAiLCAiKQp9CgpmdW5jIEdldFVwc2VydEFyZ3MobSBTdHJ1Y3RUbXBsRGF0YSkgc3RyaW5nIHsKCXZhciBwYXJ0cyBbXXN0cmluZwoJZm9yIF8sIGZsIDo9IHJhbmdlIG0uTW9kZWwuRmllbGRzIHsKCQlzd2l0Y2ggZmwuTmFtZSB7CgkJY2FzZSAiQ3JlYXRlZEF0IjoKCQkJY29udGludWUKCQl9CgkJcGFydHMgPSBhcHBlbmQocGFydHMsIGZtdC5TcHJpbnRmKCIlcy4lcyIsIG0uUmVjZWl2ZXIsIGZsLk5hbWUpKQoJfQoJcmV0dXJuIHN0cmluZ3MuSm9pbihwYXJ0cywgIiwgIikKfQ==\"") packr.PackJSONBytes("./tmpl", "types.go", "\"cGFja2FnZSB0bXBsCgovLyBUbXBsU3RydWN0cyBpcyBhIGNvbGxlY3Rpb24gb24gVG1wbFN0cnVjdAp0eXBlIFRtcGxTdHJ1Y3RzIFtdVG1wbFN0cnVjdAoKLy8gVG1wbFN0cnVjdCBkZWZpbmVzIHRoZSB0YWJsZSBkYXRhIHRvIHBhc3MgdG8gdGhlIG1vZGVscwp0eXBlIFRtcGxTdHJ1Y3Qgc3RydWN0IHsKCU5hbWUgICAgICBzdHJpbmcKCVRhYmxlTmFtZSBzdHJpbmcKCUZpZWxkcyAgICBbXVRtcGxGaWVsZAoJSW1wb3J0cyAgIG1hcFtzdHJpbmddc3RydWN0e30KfQoKLy8gVG1wbEZpZWxkIGRlZmluZXMgYSB0YWJsZSBmaWVsZCB0ZW1wbGF0ZQp0eXBlIFRtcGxGaWVsZCBzdHJ1Y3QgewoJTmFtZSAgICAgICBzdHJpbmcKCVR5cGUgICAgICAgc3RyaW5nCglDb2x1bW5OYW1lIHN0cmluZwoJTnVsbGFibGUgICBib29sCn0KCi8vIFN0cnVjdFRtcGxEYXRhIGRlZmluZXMgdGhlIHRvcCBsZXZlbCBzdHJ1Y3QgZGF0YSB0byBwYXNzIHRvIHRoZSBtb2RlbHMKdHlwZSBTdHJ1Y3RUbXBsRGF0YSBzdHJ1Y3QgewoJTW9kZWwgICAgICAgVG1wbFN0cnVjdAoJUmVjZWl2ZXIgICAgc3RyaW5nCglQYWNrYWdlTmFtZSBzdHJpbmcKfQo=\"") - packr.PackJSONBytes("./tmpl", "x_helpers.html", "\"{{define "helpers"}}

package {{ .PackageName }}

/*---------------------------+
| Code generated by modelgen |
|        DO NOT EDIT.        |
+---------------------------*/

import (
	"bytes"
	"context"
	"database/sql"
	"database/sql/driver"
	"encoding/json"
	"fmt"
	"log"
	"reflect"
	"strings"
	"time"

	"github.com/go-sql-driver/mysql"
)

// StdTime provides default SQL TIME format
const StdTime = "15:04:05"

// emptyTime allows default times to be considered
// null for insertion into the database.
var emptyTime = time.Time{}

// nullLiteral is helpful for checking
// for nulls, as they won't cause errors,
// yet we need the content of the file to change anyway
var nullLiteral = []byte("null")

/********
* Types *
********/

// Queryer allows sql.DB and sql.Tx to be used interchangeably, allowing you
// to use any of the model methods inside transactions or standalone calls.
type Queryer interface {
	Query(query string, args ...interface{}) (*sql.Rows, error)
	QueryRow(query string, args ...interface{}) *sql.Row
	Exec(query string, args ...interface{}) (sql.Result, error)
	BeginTx(ctx context.Context, opts *sql.TxOptions) (*sql.Tx, error)
}

/*-------------+
| Type aliases |
+-------------*/

// NullFloat64 aliases sql.NullFloat64
type NullFloat64 sql.NullFloat64

// NullString aliases sql.NullString
type NullString sql.NullString

// NullBool aliases sql.NullBool
type NullBool sql.NullBool

// NullInt64 aliases sql.NullInt64
type NullInt64 sql.NullInt64

// NullTime aliases sql.NullTime
type NullTime mysql.NullTime

// RawJSON aliases json.RawMessage
type RawJSON json.RawMessage

/*---------------------------+
| NullString implementations |
+---------------------------*/

// MarshalJSON for NullString
func (n NullString) MarshalJSON() ([]byte, error) {
	var a *string
	if n.Valid {
		a = &n.String
	}
	return json.Marshal(a)
}

// UnmarshalJSON for NullString
func (n *NullString) UnmarshalJSON(b []byte) error {
	if bytes.EqualFold(b, nullLiteral) {
		n.Valid = false
		return nil
	}
	err := json.Unmarshal(b, &n.String)
	n.Valid = err == nil
	return err
}

// Value for NullString
func (n NullString) Value() (driver.Value, error) {
	if !n.Valid {
		return nil, nil
	}
	return n.String, nil
}

// Scan for NullString
func (n *NullString) Scan(src interface{}) error {
	var a sql.NullString
	if err := a.Scan(src); err != nil {
		return err
	}
	n.String = a.String
	if reflect.TypeOf(src) != nil {
		n.Valid = true
	}
	return nil
}

/*----------------------------+
| NullFloat64 implementations |
+----------------------------*/

// MarshalJSON for NullFloat64
func (n NullFloat64) MarshalJSON() ([]byte, error) {
	var a *float64
	if n.Valid {
		a = &n.Float64
	}
	return json.Marshal(a)
}

// Value for NullFloat64
func (n NullFloat64) Value() (driver.Value, error) {
	if !n.Valid {
		return nil, nil
	}
	return n.Float64, nil
}

// UnmarshalJSON for NullFloat64
func (n *NullFloat64) UnmarshalJSON(b []byte) error {
	if bytes.EqualFold(b, nullLiteral) {
		n.Valid = false
		return nil
	}
	err := json.Unmarshal(b, &n.Float64)
	n.Valid = err == nil
	return err
}

// Scan for NullFloat64
func (n *NullFloat64) Scan(src interface{}) error {
	var a sql.NullFloat64
	if err := a.Scan(src); err != nil {
		return err
	}
	n.Float64 = a.Float64
	if reflect.TypeOf(src) != nil {
		n.Valid = true
	}
	return nil
}

/*--------------------------+
| NullInt64 implementations |
+--------------------------*/

// MarshalJSON for NullInt64
func (n NullInt64) MarshalJSON() ([]byte, error) {
	var a *int64
	if n.Valid {
		a = &n.Int64
	}
	return json.Marshal(a)
}

// Value for NullInt64
func (n NullInt64) Value() (driver.Value, error) {
	if !n.Valid {
		return nil, nil
	}
	return n.Int64, nil
}

// UnmarshalJSON for NullInt64
func (n *NullInt64) UnmarshalJSON(b []byte) error {
	if bytes.EqualFold(b, nullLiteral) {
		n.Valid = false
		return nil
	}
	err := json.Unmarshal(b, &n.Int64)
	n.Valid = err == nil
	return err
}

// Scan for NullInt64
func (n *NullInt64) Scan(src interface{}) error {
	var a sql.NullInt64
	if err := a.Scan(src); err != nil {
		return err
	}
	n.Int64 = a.Int64
	if reflect.TypeOf(src) != nil {
		n.Valid = true
	}
	return nil
}

/*-------------------------+
| NullBool implementations |
+-------------------------*/

// MarshalJSON for NullBool
func (n NullBool) MarshalJSON() ([]byte, error) {
	var a *bool
	if n.Valid {
		a = &n.Bool
	}
	return json.Marshal(a)
}

// Value for NullBool
func (n NullBool) Value() (driver.Value, error) {
	if !n.Valid {
		return nil, nil
	}
	return n.Bool, nil
}

// UnmarshalJSON for NullBool
func (n *NullBool) UnmarshalJSON(b []byte) error {
	var field *bool
	err := json.Unmarshal(b, &field)
	if field != nil {
		n.Valid = true
		n.Bool = *field
	}
	return err
}

// Scan for NullBool
func (n *NullBool) Scan(src interface{}) error {
	var a sql.NullBool
	if err := a.Scan(src); err != nil {
		return err
	}
	n.Bool = a.Bool
	if reflect.TypeOf(src) != nil {
		n.Valid = true
	}
	return nil
}

/*-------------------------+
| NullTime implementations |
+-------------------------*/

// MarshalJSON for NullTime
func (n NullTime) MarshalJSON() ([]byte, error) {
	var a *time.Time
	if n.Valid {
		a = &n.Time
	}
	return json.Marshal(a)
}

// Value for NullTime
func (n NullTime) Value() (driver.Value, error) {
	if !n.Valid {
		return nil, nil
	}
	return n.Time, nil
}

// UnmarshalJSON for NullTime
func (n *NullTime) UnmarshalJSON(b []byte) error {
    s := string(b)
    s = strings.Trim(s, `"`)

    var (
        zeroTime time.Time
        tim      time.Time
        err      error
    )

    if strings.EqualFold(s, "null") {
        return nil
    }

    if tim, err = time.Parse(time.RFC3339, s); err != nil {
        n.Valid = false
        return err
    }

    if tim == zeroTime {
        return nil
    }

    n.Time = tim
    n.Valid = true
    return nil
}

// Scan for NullTime
func (n *NullTime) Scan(src interface{}) error {
	// Set initial state for subsequent scans.
	n.Valid = false

	var a mysql.NullTime
	if err := a.Scan(src); err != nil {
		return err
	}
	n.Time = a.Time
	if reflect.TypeOf(src) != nil {
		n.Valid = true
	}
	return nil
}

/*------------------------+
| RawJSON implementations |
+------------------------*/

// MarshalJSON for NullString
func (n RawJSON) MarshalJSON() ([]byte, error) {
	a := json.RawMessage(n)
	return a.MarshalJSON()
}

// Value for NullString
func (n RawJSON) Value() (driver.Value, error) {
	return string(n), nil
}

// UnmarshalJSON for NullString
func (n *RawJSON) UnmarshalJSON(b []byte) error {
	var a json.RawMessage
	if err := json.Unmarshal(b, &a); err != nil {
		return err
	}
	c := RawJSON(a)
	*n = c
	return nil
}

// Scan for NullString
func (n *RawJSON) Scan(src interface{}) error {
	var a sql.NullString
	if err := a.Scan(src); err != nil {
		return err
	}
	jsn := RawJSON([]byte(a.String))
	*n = jsn
	return nil
}

/*-----------------+
| Helper functions |
+-----------------*/

// ToNullString returns a new NullString
func ToNullString(s *string) NullString {
	if s == nil {
		return NullString(sql.NullString{Valid: false})
	}
	return NullString(sql.NullString{String: *s, Valid: true})
}

// ToNullInt64 returns a new NullInt64
func ToNullInt64(i *int64) NullInt64 {
	if i == nil {
		return NullInt64(sql.NullInt64{Valid: false})
	}
	return NullInt64(sql.NullInt64{Int64: *i, Valid: true})
}

// ToNullFloat64 returns a new NullFloat64
func ToNullFloat64(i *float64) NullFloat64 {
	if i == nil {
		return NullFloat64(sql.NullFloat64{Valid: false})
	}
	return NullFloat64(sql.NullFloat64{Float64: *i, Valid: true})
}

// ToNullBool creates a new NullBool
func ToNullBool(b *bool) NullBool {
	if b == nil {
		return NullBool(sql.NullBool{Valid: false})
	}
	return NullBool(sql.NullBool{Bool: *b, Valid: true})
}

// ToNullTime creates a new NullTime
func ToNullTime(t time.Time) NullTime {
	if t == emptyTime {
		return NullTime(mysql.NullTime{Valid: false})
	}
	return NullTime(mysql.NullTime{Time: t, Valid: true})
}

// TxOptions defines an option type for configuring
// transations. This may only be used with the ExecuteTransaction wrapper.
type TxOptions struct {
	Timeout   time.Duration
	Isolation sql.IsolationLevel
	ReadOnly  bool
}

// ExecuteTransaction closes over a transaction and automatically commits
// or rollbacks depending on whether errors were encountered.
// In the case where nil is passed for opt (*TxOption), the following defaults are used:
//  &TxOptions{
//  	Timeout:   5 * time.Second,
//  	Isolation: sql.LevelSerializable,
//  	ReadOnly:  false,
//  }
func ExecuteTransaction(qu Queryer, opt *TxOptions, actions func(*sql.Tx) error) (err error) {
	// Provide safe defaults in case none were given.
	if opt == nil {
		opt = &TxOptions{
			Timeout:   5 * time.Second,
			Isolation: sql.LevelSerializable,
			ReadOnly:  false,
		}
	}

	// Build the context with the provided timeout.
	// This will be used to define the total time the transaction may take,
	// past this time, it will be cancelled, rollback, then throw an error.
	ctx, cancel := context.WithTimeout(context.Background(), opt.Timeout)
	defer cancel()

	var tx *sql.Tx
	if tx, err = qu.BeginTx(ctx, &sql.TxOptions{
		Isolation: opt.Isolation,
		ReadOnly:  opt.ReadOnly,
	}); err != nil {
		return err
	}

	defer func() {
		if r := recover(); r != nil {
			// Only need to log here because panic won't report whether
			// the rollback was successful or not.
			if txerr := tx.Rollback(); txerr != nil {
				log.Println("db rollback error:", txerr)
			}

			log.Printf("rolled back transaction")
			panic(r)
		} else if err != nil {
			// If we run into issues rolling back, keep track of the error that
			// caused the issue and provide some context on the rollback failure.
			if rerr := tx.Rollback(); rerr != nil {
				err = fmt.Errorf("db error: %v rollback error: %v", err, rerr)
			}
		} else {
			if cerr := tx.Commit(); cerr != nil {
				err = fmt.Errorf("commit error: %v", cerr)
			}
		}
	}()

	err = actions(tx)
	return err
}
{{end}}
\"") - packr.PackJSONBytes("./tmpl", "x_helpers_test.html", "\"{{define "helperstest"}}
//+build !helpers

package {{ .PackageName }}

/*---------------------------+
| Code generated by modelgen |
|        DO NOT EDIT.        |
+---------------------------*/

import (
	"database/sql/driver"
	"encoding/json"
	"reflect"
	"testing"
	"time"
)

func TestStructEmbedding(t *testing.T) {
	tim := time.Date(2017, 1, 1, 0, 0, 0, 0, time.UTC)
	expected := []byte(`{"a":123,"b":true,"c":123.123,"d":"string","e":"2017-01-01T00:00:00Z","f":[1,2,3]}`)
	type embed struct {
		A NullInt64   `json:"a,omitempty"`
		B NullBool    `json:"b,omitempty"`
		C NullFloat64 `json:"c,omitempty"`
		D NullString  `json:"d,omitempty"`
		E NullTime    `json:"e,omitempty"`
		F RawJSON     `json:"f,omitempty"`
	}
	em := embed{
		A: NullInt64{Valid: true, Int64: 123},
		B: NullBool{Valid: true, Bool: true},
		C: NullFloat64{Valid: true, Float64: 123.123},
		D: NullString{Valid: true, String: "string"},
		E: NullTime{Valid: true, Time: tim},
		F: RawJSON(`[1,2,3]`),
	}
	b, err := json.Marshal(em)
	if err != nil {
		t.Fatal(err)
	}
	if !reflect.DeepEqual(expected, b) {
		t.Fatal("not the same JSON!")
	}
	if !(string(b) == string(expected)) {
		t.Fatal("not the same!")
	}

	var em2 embed
	if err := json.Unmarshal(expected, &em2); err != nil {
		t.Fatal(err)
	}
	if !reflect.DeepEqual(em2, em) {
		t.Fatal("not correct")
	}
}

func TestNullString_UnmarshalJSON(t *testing.T) {
	tests := []struct {
		name    string
		n       NullString
		source  []byte
		wantErr bool
	}{
		{
			name:    "explicit null",
			source:  []byte(`null`),
			wantErr: false,
		},
		{
			name:    "string null",
			source:  []byte(`"null"`),
			wantErr: false, // this one SHOULD be valid
		},
		{
			name:    "valid",
			source:  []byte(`"hello"`),
			wantErr: false,
		},
		{
			name:    "invalid",
			source:  []byte(`{"key":"value"}`),
			wantErr: true,
		},
		{
			name:    "empty",
			source:  []byte{},
			wantErr: true,
		},
	}
	for _, tt := range tests {
		t.Run(tt.name, func(t *testing.T) {
			if err := tt.n.UnmarshalJSON(tt.source); (err != nil) != tt.wantErr {
				t.Errorf("NullString.UnmarshalJSON() error = %v, wantErr %v", err, tt.wantErr)
			}
		})
	}
}

func TestNullString_Value(t *testing.T) {
	tests := []struct {
		name    string
		n       NullString
		want    driver.Value
		wantErr bool
	}{
		{
			name: "valid",
			n: NullString{
				Valid:  true,
				String: "hello",
			},
			want:    driver.Value("hello"),
			wantErr: false,
		},
		{
			name: "invalid",
			n: NullString{
				Valid: false,
			},
			want:    nil,
			wantErr: false,
		},
	}
	for _, tt := range tests {
		t.Run(tt.name, func(t *testing.T) {
			got, err := tt.n.Value()
			if (err != nil) != tt.wantErr {
				t.Errorf("NullString.Value() error = %v, wantErr %v", err, tt.wantErr)
				return
			}
			if !reflect.DeepEqual(got, tt.want) {
				t.Errorf("NullString.Value() = %v, want %v", got, tt.want)
			}
		})
	}
}

func TestNullString_Scan(t *testing.T) {
	tests := []struct {
		name    string
		n       *NullString
		wantErr bool
		src     interface{}
	}{
		{
			name: "valid",
			n: &NullString{
				String: "hello",
				Valid:  true,
			},
			src:     "",
			wantErr: false,
		},
		{
			name: "nil value",
			n: &NullString{
				String: "hello",
				Valid:  false,
			},
			src:     nil,
			wantErr: false,
		},
	}
	for _, tt := range tests {
		t.Run(tt.name, func(t *testing.T) {
			if err := tt.n.Scan(tt.src); (err != nil) != tt.wantErr {
				t.Errorf("NullString.Scan() error = %v, wantErr %v", err, tt.wantErr)
			}
			if !tt.n.Valid && tt.src != nil {
				t.Errorf("should return null")
			}
			if tt.n.Valid && tt.src != tt.n.String {
				t.Errorf("invalid value")
			}
		})
	}
}

func TestNullString_MarshalJSON(t *testing.T) {
	tests := []struct {
		name    string
		n       *NullString
		want    []byte
		wantErr bool
	}{
		{
			name: "valid",
			n: &NullString{
				String: "hello",
				Valid:  true,
			},
			want:    []byte(`"hello"`),
			wantErr: false,
		},
		{
			name: "valid null",
			n: &NullString{
				String: "",
				Valid:  false,
			},
			want:    []byte(`null`),
			wantErr: false,
		},
		{
			name: "invalid",
			n: &NullString{
				Valid: true,
			},
			want:    []byte(`""`),
			wantErr: false,
		},
	}
	for _, tt := range tests {
		t.Run(tt.name, func(t *testing.T) {
			got, err := tt.n.MarshalJSON()
			if (err != nil) != tt.wantErr {
				t.Errorf("NullString.MarshalJSON() error = %v, wantErr %v", err, tt.wantErr)
				return
			}
			if !reflect.DeepEqual(got, tt.want) {
				t.Errorf("NullString.MarshalJSON() = %v, want %v", got, tt.want)
			}
		})
	}
}

func TestNullBool_UnmarshalJSON(t *testing.T) {
	tests := []struct {
		name         string
		n            NullBool
		source       []byte
		wantErr      bool
		wantValidity bool
	}{
		{
			name:         "explicit null",
			source:       []byte(`null`),
			wantErr:      false,
			wantValidity: false,
		},
		{
			name:         "valid",
			source:       []byte(`false`),
			wantErr:      false,
			wantValidity: true,
		},
		{
			name:         "invalid",
			source:       []byte(`{"key":"value"}`),
			wantErr:      true,
			wantValidity: false,
		},
		{
			name:         "empty",
			source:       []byte{},
			wantErr:      true,
			wantValidity: false,
		},
		{
			name:         "explicit null",
			source:       []byte("null"),
			wantErr:      false,
			wantValidity: false,
		},
	}
	for _, tt := range tests {
		t.Run(tt.name, func(t *testing.T) {
			if err := tt.n.UnmarshalJSON(tt.source); (err != nil) != tt.wantErr && tt.n.Valid == tt.wantValidity {
				t.Errorf("NullBool.UnmarshalJSON() error = %v, wantErr %v", err, tt.wantErr)
			}
		})
	}
}

func TestNullBool_Value(t *testing.T) {
	tests := []struct {
		name    string
		n       NullBool
		want    driver.Value
		wantErr bool
	}{
		{
			name: "valid",
			n: NullBool{
				Valid: true,
				Bool:  true,
			},
			want:    driver.Value(true),
			wantErr: false,
		},
		{
			name: "invalid",
			n: NullBool{
				Valid: false,
			},
			want:    nil,
			wantErr: false,
		},
	}
	for _, tt := range tests {
		t.Run(tt.name, func(t *testing.T) {
			got, err := tt.n.Value()
			if (err != nil) != tt.wantErr {
				t.Errorf("NullBool.Value() error = %v, wantErr %v", err, tt.wantErr)
				return
			}
			if !reflect.DeepEqual(got, tt.want) {
				t.Errorf("NullBool.Value() = %v, want %v", got, tt.want)
			}
		})
	}
}

func TestNullBool_Scan(t *testing.T) {
	tests := []struct {
		name    string
		n       *NullBool
		wantErr bool
		src     interface{}
	}{
		{
			name: "valid",
			n: &NullBool{
				Bool:  true,
				Valid: true,
			},
			src:     true,
			wantErr: false,
		},
		{
			name: "nil value",
			n: &NullBool{
				Bool:  true,
				Valid: false,
			},
			src:     false,
			wantErr: false,
		},
	}
	for _, tt := range tests {
		t.Run(tt.name, func(t *testing.T) {
			if err := tt.n.Scan(tt.src); (err != nil) != tt.wantErr {
				t.Errorf("NullBool.Scan() error = %v, wantErr %v", err, tt.wantErr)
			}
			if !tt.n.Valid && tt.src != nil {
				t.Errorf("should return null")
			}
			if tt.n.Valid && tt.src != tt.n.Bool {
				t.Errorf("invalid value")
			}
		})
	}
}

func TestNullBool_MarshalJSON(t *testing.T) {
	tests := []struct {
		name    string
		n       *NullBool
		want    []byte
		wantErr bool
	}{
		{
			name: "valid",
			n: &NullBool{
				Valid: true,
			},
			want:    []byte(`false`),
			wantErr: false,
		},
		{
			name: "valid null",
			n: &NullBool{
				Valid: false,
			},
			want:    []byte(`null`),
			wantErr: false,
		},
		{
			name: "invalid",
			n: &NullBool{
				Valid: true,
			},
			want:    []byte(`false`),
			wantErr: false,
		},
	}
	for _, tt := range tests {
		t.Run(tt.name, func(t *testing.T) {
			got, err := tt.n.MarshalJSON()
			if (err != nil) != tt.wantErr {
				t.Errorf("NullBool.MarshalJSON() error = %v, wantErr %v", err, tt.wantErr)
				return
			}
			if !reflect.DeepEqual(got, tt.want) {
				t.Errorf("NullBool.MarshalJSON() = %v, want %v", got, tt.want)
			}
		})
	}
}

func TestNullTime_UnmarshalJSON(t *testing.T) {
	tests := []struct {
		name    string
		n       NullTime
		source  []byte
		wantErr bool
	}{
		{
			name:    "valid",
			source:  []byte(`"2017-11-24T00:00:00Z"`),
			wantErr: false,
		},
		{
			name:    "invalid",
			source:  []byte(`{"key":"value"}`),
			wantErr: true,
		},
		{
			name:    "empty",
			source:  []byte{},
			wantErr: true,
		},
		{
			name:    "explicit null",
			source:  []byte(`null`),
			wantErr: false,
		},
	}
	for _, tt := range tests {
		t.Run(tt.name, func(t *testing.T) {
			if err := tt.n.UnmarshalJSON(tt.source); (err != nil) != tt.wantErr {
				t.Errorf("NullBool.UnmarshalJSON() error = %v, wantErr %v", err, tt.wantErr)
			}
		})
	}
}

func TestNullTime_Value(t *testing.T) {
	tim := time.Now()
	tests := []struct {
		name    string
		n       NullTime
		want    driver.Value
		wantErr bool
	}{
		{
			name: "valid",
			n: NullTime{
				Valid: true,
				Time:  tim,
			},
			want:    driver.Value(tim),
			wantErr: false,
		},
		{
			name: "invalid",
			n: NullTime{
				Valid: false,
			},
			want:    nil,
			wantErr: false,
		},
	}
	for _, tt := range tests {
		t.Run(tt.name, func(t *testing.T) {
			got, err := tt.n.Value()
			if (err != nil) != tt.wantErr {
				t.Errorf("NullTime.Value() error = %v, wantErr %v", err, tt.wantErr)
				return
			}
			if !reflect.DeepEqual(got, tt.want) {
				t.Errorf("NullTime.Value() = %v, want %v", got, tt.want)
			}
		})
	}
}

func TestNullTime_Scan(t *testing.T) {
	tim := time.Now()
	tests := []struct {
		name    string
		n       *NullTime
		wantErr bool
		src     interface{}
	}{
		{
			name: "valid",
			n: &NullTime{
				Time:  tim,
				Valid: true,
			},
			src:     tim,
			wantErr: false,
		},
		{
			name: "nil value",
			n: &NullTime{
				Time:  tim,
				Valid: false,
			},
			src:     time.Now(),
			wantErr: false,
		},
	}
	for _, tt := range tests {
		t.Run(tt.name, func(t *testing.T) {
			if err := tt.n.Scan(tt.src); (err != nil) != tt.wantErr {
				t.Errorf("NullTime.Scan() error = %v, wantErr %v", err, tt.wantErr)
			}
			if !tt.n.Valid && tt.src != nil {
				t.Errorf("should return null")
			}
			if tt.n.Valid && tt.src != tt.n.Time {
				t.Errorf("invalid value")
			}
		})
	}
}

func TestNullTime_MarshalJSON(t *testing.T) {
	tests := []struct {
		name    string
		n       *NullTime
		want    []byte
		wantErr bool
	}{
		{
			name: "valid",
			n: &NullTime{
				Time:  time.Date(2017, 11, 24, 0, 0, 0, 0, time.UTC),
				Valid: true,
			},
			want:    []byte(`"2017-11-24T00:00:00Z"`),
			wantErr: false,
		},
		{
			name: "valid null",
			n: &NullTime{
				Valid: false,
			},
			want:    []byte(`null`),
			wantErr: false,
		},
		{
			name: "invalid",
			n: &NullTime{
				Valid: true,
			},
			want:    []byte(`"0001-01-01T00:00:00Z"`),
			wantErr: false,
		},
	}
	for _, tt := range tests {
		t.Run(tt.name, func(t *testing.T) {
			got, err := tt.n.MarshalJSON()
			if (err != nil) != tt.wantErr {
				t.Errorf("NullTime.MarshalJSON() error = %v, wantErr %v", err, tt.wantErr)
				return
			}
			if !reflect.DeepEqual(got, tt.want) {
				t.Errorf("NullTime.MarshalJSON() = %v, want %v", got, tt.want)
			}
		})
	}
}

func TestNullInt64_UnmarshalJSON(t *testing.T) {
	tests := []struct {
		name    string
		n       NullInt64
		source  []byte
		wantErr bool
	}{
		{
			name:    "valid",
			source:  []byte(`123`),
			wantErr: false,
		},
		{
			name:    "invalid",
			source:  []byte(`{"key":"value"}`),
			wantErr: true,
		},
		{
			name:    "empty",
			source:  []byte{},
			wantErr: true,
		},
	}
	for _, tt := range tests {
		t.Run(tt.name, func(t *testing.T) {
			if err := tt.n.UnmarshalJSON(tt.source); (err != nil) != tt.wantErr {
				t.Errorf("NullInt64.UnmarshalJSON() error = %v, wantErr %v", err, tt.wantErr)
			}
		})
	}
}

func TestNullInt64_Value(t *testing.T) {
	tests := []struct {
		name    string
		n       NullInt64
		want    driver.Value
		wantErr bool
	}{
		{
			name: "valid",
			n: NullInt64{
				Valid: true,
				Int64: 123,
			},
			want:    driver.Value(int64(123)),
			wantErr: false,
		},
		{
			name: "invalid",
			n: NullInt64{
				Valid: false,
			},
			want:    nil,
			wantErr: false,
		},
	}
	for _, tt := range tests {
		t.Run(tt.name, func(t *testing.T) {
			got, err := tt.n.Value()
			if (err != nil) != tt.wantErr {
				t.Errorf("NullInt64.Value() error = %v, wantErr %v", err, tt.wantErr)
				return
			}
			if !reflect.DeepEqual(got, tt.want) {
				t.Errorf("NullInt64.Value() = %v, want %v", got, tt.want)
			}
		})
	}
}

func TestNullInt64_Scan(t *testing.T) {
	tests := []struct {
		name    string
		n       *NullInt64
		wantErr bool
		src     interface{}
	}{
		{
			name: "valid",
			n: &NullInt64{
				Int64: 123,
				Valid: true,
			},
			src:     int64(123),
			wantErr: false,
		},
		{
			name: "nil value",
			n: &NullInt64{
				Valid: false,
			},
			src:     int64(123),
			wantErr: false,
		},
	}
	for _, tt := range tests {
		t.Run(tt.name, func(t *testing.T) {
			if err := tt.n.Scan(tt.src); (err != nil) != tt.wantErr {
				t.Errorf("NullInt64.Scan() error = %v, wantErr %v", err, tt.wantErr)
			}
			if !tt.n.Valid && tt.src != nil {
				t.Errorf("should return null")
			}
			if tt.n.Valid && tt.src != tt.n.Int64 {
				t.Errorf("invalid value")
			}
		})
	}
}

func TestNullInt64_MarshalJSON(t *testing.T) {
	tests := []struct {
		name    string
		n       *NullInt64
		want    []byte
		wantErr bool
	}{
		{
			name: "valid",
			n: &NullInt64{
				Int64: 123,
				Valid: true,
			},
			want:    []byte(`123`),
			wantErr: false,
		},
		{
			name: "valid null",
			n: &NullInt64{
				Valid: false,
			},
			want:    []byte(`null`),
			wantErr: false,
		},
		{
			name: "invalid",
			n: &NullInt64{
				Valid: true,
			},
			want:    []byte(`0`),
			wantErr: false,
		},
	}
	for _, tt := range tests {
		t.Run(tt.name, func(t *testing.T) {
			got, err := tt.n.MarshalJSON()
			if (err != nil) != tt.wantErr {
				t.Errorf("NullInt64.MarshalJSON() error = %v, wantErr %v", err, tt.wantErr)
				return
			}
			if !reflect.DeepEqual(got, tt.want) {
				t.Errorf("NullInt64.MarshalJSON() = %v, want %v", got, tt.want)
			}
		})
	}
}
func TestNullFloat64_UnmarshalJSON(t *testing.T) {
	tests := []struct {
		name    string
		n       NullFloat64
		source  []byte
		wantErr bool
	}{
		{
			name:    "explicit null",
			source:  []byte(`null`),
			wantErr: false,
		},
		{
			name:    "valid",
			source:  []byte(`123.123`),
			wantErr: false,
		},
		{
			name:    "invalid",
			source:  []byte(`{"key":"value"}`),
			wantErr: true,
		},
		{
			name:    "empty",
			source:  []byte{},
			wantErr: true,
		},
	}
	for _, tt := range tests {
		t.Run(tt.name, func(t *testing.T) {
			if err := tt.n.UnmarshalJSON(tt.source); (err != nil) != tt.wantErr {
				t.Errorf("NullFloat64.UnmarshalJSON() error = %v, wantErr %v", err, tt.wantErr)
			}
		})
	}
}

func TestNullFloat64_Value(t *testing.T) {
	tests := []struct {
		name    string
		n       NullFloat64
		want    driver.Value
		wantErr bool
	}{
		{
			name: "valid",
			n: NullFloat64{
				Valid:   true,
				Float64: 123.123,
			},
			want:    driver.Value(float64(123.123)),
			wantErr: false,
		},
		{
			name: "invalid",
			n: NullFloat64{
				Valid: false,
			},
			want:    nil,
			wantErr: false,
		},
	}
	for _, tt := range tests {
		t.Run(tt.name, func(t *testing.T) {
			got, err := tt.n.Value()
			if (err != nil) != tt.wantErr {
				t.Errorf("NullFloat64.Value() error = %v, wantErr %v", err, tt.wantErr)
				return
			}
			if !reflect.DeepEqual(got, tt.want) {
				t.Errorf("NullFloat64.Value() = %v, want %v", got, tt.want)
			}
		})
	}
}

func TestNullFloat64_Scan(t *testing.T) {
	tests := []struct {
		name    string
		n       *NullFloat64
		wantErr bool
		src     interface{}
	}{
		{
			name: "valid",
			n: &NullFloat64{
				Float64: 123.123,
				Valid:   true,
			},
			src:     float64(123),
			wantErr: false,
		},
		{
			name: "nil value",
			n: &NullFloat64{
				Valid: false,
			},
			src:     float64(123.123),
			wantErr: false,
		},
	}
	for _, tt := range tests {
		t.Run(tt.name, func(t *testing.T) {
			if err := tt.n.Scan(tt.src); (err != nil) != tt.wantErr {
				t.Errorf("NullFloat64.Scan() error = %v, wantErr %v", err, tt.wantErr)
			}
			if !tt.n.Valid && tt.src != nil {
				t.Errorf("should return null")
			}
			if tt.n.Valid && tt.src != tt.n.Float64 {
				t.Errorf("invalid value")
			}
		})
	}
}

func TestNullFloat64_MarshalJSON(t *testing.T) {
	tests := []struct {
		name    string
		n       *NullFloat64
		want    []byte
		wantErr bool
	}{
		{
			name: "valid",
			n: &NullFloat64{
				Float64: 123.123,
				Valid:   true,
			},
			want:    []byte(`123.123`),
			wantErr: false,
		},
		{
			name: "valid null",
			n: &NullFloat64{
				Valid: false,
			},
			want:    []byte(`null`),
			wantErr: false,
		},
		{
			name: "invalid",
			n: &NullFloat64{
				Valid: true,
			},
			want:    []byte(`0`),
			wantErr: false,
		},
	}
	for _, tt := range tests {
		t.Run(tt.name, func(t *testing.T) {
			got, err := tt.n.MarshalJSON()
			if (err != nil) != tt.wantErr {
				t.Errorf("NullFloat64.MarshalJSON() error = %v, wantErr %v", err, tt.wantErr)
				return
			}
			if !reflect.DeepEqual(got, tt.want) {
				t.Errorf("NullFloat64.MarshalJSON() = %v, want %v", got, tt.want)
			}
		})
	}
}

func TestToNullBool(t *testing.T) {
	b := true
	bb := ToNullBool(&b)
	if !bb.Valid {
		t.Errorf("expected valid, got %v", bb.Valid)
	}
	if !bb.Bool {
		t.Errorf("expected true, got %v", bb.Bool)
	}

	var b2 *bool
	bb2 := ToNullBool(b2)
	if bb2.Valid {
		t.Errorf("expected not valid, got %v", bb2.Valid)
	}
	if bb2.Bool {
		t.Errorf("expected false, got %v", bb2.Bool)
	}
}
func TestToNullInt64(t *testing.T) {
	b := int64(123)
	bb := ToNullInt64(&b)
	if !bb.Valid {
		t.Errorf("expected valid, got %v", bb.Valid)
	}
	if bb.Int64 != 123 {
		t.Errorf("expected 123, got %v", bb.Int64)
	}

	var b2 *int64
	bb2 := ToNullInt64(b2)
	if bb2.Valid {
		t.Errorf("expected not valid, got %v", bb2.Valid)
	}
	if bb2.Int64 != 0 {
		t.Errorf("expected 0, got %v", bb2.Int64)
	}
}

func TestToNullFloat64(t *testing.T) {
	b := float64(123.123)
	bb := ToNullFloat64(&b)
	if !bb.Valid {
		t.Errorf("expected valid, got %v", bb.Valid)
	}
	if bb.Float64 != 123.123 {
		t.Errorf("expected 123.123, got %v", bb.Float64)
	}

	var b2 *float64
	bb2 := ToNullFloat64(b2)
	if bb2.Valid {
		t.Errorf("expected not valid, got %v", bb2.Valid)
	}
	if bb2.Float64 != 0 {
		t.Errorf("expected 0, got %v", bb2.Float64)
	}
}
func TestToNullString(t *testing.T) {
	b := "qwe"
	bb := ToNullString(&b)
	if !bb.Valid {
		t.Errorf("expected valid, got %v", bb.Valid)
	}
	if bb.String != "qwe" {
		t.Errorf("expected qwe, got %v", bb.String)
	}

	var b2 *string
	bb2 := ToNullString(b2)
	if bb2.Valid {
		t.Errorf("expected not valid, got %v", bb2.Valid)
	}
	if bb2.String != "" {
		t.Errorf("expected <empty string>, got %v", bb2.String)
	}
}
func TestToNullTime(t *testing.T) {
	tim := time.Now()
	bb := ToNullTime(tim)
	if !bb.Valid {
		t.Errorf("expected valid, got %v", bb.Valid)
	}
	if bb.Time != tim {
		t.Errorf("expected %v, got %v", tim, bb.Time)
	}

	tim = time.Time{}
	bb = ToNullTime(tim)
	if bb.Valid {
		t.Errorf("expected invalid, got %v", bb.Valid)
	}
	if bb.Time != tim {
		t.Errorf("expected %v, got %v", tim, bb.Time)
	}
}
{{end}}
\"") + packr.PackJSONBytes("./tmpl", "x_helpers.html", "\"{{define "helpers"}}

package {{ .PackageName }}

/*---------------------------+
| Code generated by modelgen |
|        DO NOT EDIT.        |
+---------------------------*/

import (
	"bytes"
	"context"
	"database/sql"
	"database/sql/driver"
	"encoding/json"
	"fmt"
	"log"
	"reflect"
	"strings"
	"time"

	"github.com/go-sql-driver/mysql"
)

// StdTime provides default SQL TIME format
const StdTime = "15:04:05"

// emptyTime allows default times to be considered
// null for insertion into the database.
var emptyTime = time.Time{}

// nullLiteral is helpful for checking
// for nulls, as they won't cause errors,
// yet we need the content of the file to change anyway
var nullLiteral = []byte("null")

/********
* Types *
********/

// Queryer allows sql.DB and sql.Tx to be used interchangeably, allowing you
// to use any of the model methods inside transactions or standalone calls.
type Queryer interface {
	Query(query string, args ...interface{}) (*sql.Rows, error)
	QueryRow(query string, args ...interface{}) *sql.Row
	Exec(query string, args ...interface{}) (sql.Result, error)
	BeginTx(ctx context.Context, opts *sql.TxOptions) (*sql.Tx, error)
}

/*-------------+
| Type aliases |
+-------------*/

// NullFloat64 aliases sql.NullFloat64
type NullFloat64 sql.NullFloat64

// NullString aliases sql.NullString
type NullString sql.NullString

// NullBool aliases sql.NullBool
type NullBool sql.NullBool

// NullInt64 aliases sql.NullInt64
type NullInt64 sql.NullInt64

// NullTime aliases sql.NullTime
type NullTime mysql.NullTime

// RawJSON aliases json.RawMessage
type RawJSON json.RawMessage

/*---------------------------+
| NullString implementations |
+---------------------------*/

// MarshalJSON for NullString
func (n NullString) MarshalJSON() ([]byte, error) {
	var a *string
	if n.Valid {
		a = &n.String
	}
	return json.Marshal(a)
}

// UnmarshalJSON for NullString
func (n *NullString) UnmarshalJSON(b []byte) error {
	if bytes.EqualFold(b, nullLiteral) {
		n.Valid = false
		return nil
	}
	err := json.Unmarshal(b, &n.String)
	n.Valid = err == nil
	return err
}

// Value for NullString
func (n NullString) Value() (driver.Value, error) {
	if !n.Valid {
		return nil, nil
	}
	return n.String, nil
}

// Scan for NullString
func (n *NullString) Scan(src interface{}) error {
	var a sql.NullString
	if err := a.Scan(src); err != nil {
		return err
	}
	n.String = a.String
	if reflect.TypeOf(src) != nil {
		n.Valid = true
	}
	return nil
}

/*----------------------------+
| NullFloat64 implementations |
+----------------------------*/

// MarshalJSON for NullFloat64
func (n NullFloat64) MarshalJSON() ([]byte, error) {
	var a *float64
	if n.Valid {
		a = &n.Float64
	}
	return json.Marshal(a)
}

// Value for NullFloat64
func (n NullFloat64) Value() (driver.Value, error) {
	if !n.Valid {
		return nil, nil
	}
	return n.Float64, nil
}

// UnmarshalJSON for NullFloat64
func (n *NullFloat64) UnmarshalJSON(b []byte) error {
	if bytes.EqualFold(b, nullLiteral) {
		n.Valid = false
		return nil
	}
	err := json.Unmarshal(b, &n.Float64)
	n.Valid = err == nil
	return err
}

// Scan for NullFloat64
func (n *NullFloat64) Scan(src interface{}) error {
	var a sql.NullFloat64
	if err := a.Scan(src); err != nil {
		return err
	}
	n.Float64 = a.Float64
	if reflect.TypeOf(src) != nil {
		n.Valid = true
	}
	return nil
}

/*--------------------------+
| NullInt64 implementations |
+--------------------------*/

// MarshalJSON for NullInt64
func (n NullInt64) MarshalJSON() ([]byte, error) {
	var a *int64
	if n.Valid {
		a = &n.Int64
	}
	return json.Marshal(a)
}

// Value for NullInt64
func (n NullInt64) Value() (driver.Value, error) {
	if !n.Valid {
		return nil, nil
	}
	return n.Int64, nil
}

// UnmarshalJSON for NullInt64
func (n *NullInt64) UnmarshalJSON(b []byte) error {
	if bytes.EqualFold(b, nullLiteral) {
		n.Valid = false
		return nil
	}
	err := json.Unmarshal(b, &n.Int64)
	n.Valid = err == nil
	return err
}

// Scan for NullInt64
func (n *NullInt64) Scan(src interface{}) error {
	var a sql.NullInt64
	if err := a.Scan(src); err != nil {
		return err
	}
	n.Int64 = a.Int64
	if reflect.TypeOf(src) != nil {
		n.Valid = true
	}
	return nil
}

/*-------------------------+
| NullBool implementations |
+-------------------------*/

// MarshalJSON for NullBool
func (n NullBool) MarshalJSON() ([]byte, error) {
	var a *bool
	if n.Valid {
		a = &n.Bool
	}
	return json.Marshal(a)
}

// Value for NullBool
func (n NullBool) Value() (driver.Value, error) {
	if !n.Valid {
		return nil, nil
	}
	return n.Bool, nil
}

// UnmarshalJSON for NullBool
func (n *NullBool) UnmarshalJSON(b []byte) error {
	var field *bool
	err := json.Unmarshal(b, &field)
	if field != nil {
		n.Valid = true
		n.Bool = *field
	}
	return err
}

// Scan for NullBool
func (n *NullBool) Scan(src interface{}) error {
	var a sql.NullBool
	if err := a.Scan(src); err != nil {
		return err
	}
	n.Bool = a.Bool
	if reflect.TypeOf(src) != nil {
		n.Valid = true
	}
	return nil
}

/*-------------------------+
| NullTime implementations |
+-------------------------*/

// MarshalJSON for NullTime
func (n NullTime) MarshalJSON() ([]byte, error) {
	var a *time.Time
	if n.Valid {
		a = &n.Time
	}
	return json.Marshal(a)
}

// Value for NullTime
func (n NullTime) Value() (driver.Value, error) {
	if !n.Valid {
		return nil, nil
	}
	return n.Time, nil
}

// UnmarshalJSON for NullTime
func (n *NullTime) UnmarshalJSON(b []byte) error {
    s := string(b)
    s = strings.Trim(s, `"`)

    var (
        zeroTime time.Time
        tim      time.Time
        err      error
    )

    if strings.EqualFold(s, "null") {
        return nil
    }

    if tim, err = time.Parse(time.RFC3339, s); err != nil {
        n.Valid = false
        return err
    }

    if tim == zeroTime {
        return nil
    }

    n.Time = tim
    n.Valid = true
    return nil
}

// Scan for NullTime
func (n *NullTime) Scan(src interface{}) error {
	// Set initial state for subsequent scans.
	n.Valid = false

	var a mysql.NullTime
	if err := a.Scan(src); err != nil {
		return err
	}
	n.Time = a.Time
	if reflect.TypeOf(src) != nil {
		n.Valid = true
	}
	return nil
}

/*------------------------+
| RawJSON implementations |
+------------------------*/

// MarshalJSON for NullString
func (n RawJSON) MarshalJSON() ([]byte, error) {
	if len(n) == 0 {
		return []byte("null"), nil
	}
	a := json.RawMessage(n)
	return a.MarshalJSON()
}

// Value for NullString
func (n RawJSON) Value() (driver.Value, error) {
	return string(n), nil
}

// UnmarshalJSON for NullString
func (n *RawJSON) UnmarshalJSON(b []byte) error {
	var a json.RawMessage
	if err := json.Unmarshal(b, &a); err != nil {
		return err
	}
	c := RawJSON(a)
	*n = c
	return nil
}

// Scan for NullString
func (n *RawJSON) Scan(src interface{}) error {
	var a sql.NullString
	if err := a.Scan(src); err != nil {
		return err
	}
	jsn := RawJSON([]byte(a.String))
	*n = jsn
	return nil
}

/*-----------------+
| Helper functions |
+-----------------*/

// ToNullString returns a new NullString
func ToNullString(s *string) NullString {
	if s == nil {
		return NullString(sql.NullString{Valid: false})
	}
	return NullString(sql.NullString{String: *s, Valid: true})
}

// ToNullInt64 returns a new NullInt64
func ToNullInt64(i *int64) NullInt64 {
	if i == nil {
		return NullInt64(sql.NullInt64{Valid: false})
	}
	return NullInt64(sql.NullInt64{Int64: *i, Valid: true})
}

// ToNullFloat64 returns a new NullFloat64
func ToNullFloat64(i *float64) NullFloat64 {
	if i == nil {
		return NullFloat64(sql.NullFloat64{Valid: false})
	}
	return NullFloat64(sql.NullFloat64{Float64: *i, Valid: true})
}

// ToNullBool creates a new NullBool
func ToNullBool(b *bool) NullBool {
	if b == nil {
		return NullBool(sql.NullBool{Valid: false})
	}
	return NullBool(sql.NullBool{Bool: *b, Valid: true})
}

// ToNullTime creates a new NullTime
func ToNullTime(t time.Time) NullTime {
	if t == emptyTime {
		return NullTime(mysql.NullTime{Valid: false})
	}
	return NullTime(mysql.NullTime{Time: t, Valid: true})
}

// TxOptions defines an option type for configuring
// transations. This may only be used with the ExecuteTransaction wrapper.
type TxOptions struct {
	Timeout   time.Duration
	Isolation sql.IsolationLevel
	ReadOnly  bool
}

// ExecuteTransaction closes over a transaction and automatically commits
// or rollbacks depending on whether errors were encountered.
// In the case where nil is passed for opt (*TxOption), the following defaults are used:
//  &TxOptions{
//  	Timeout:   5 * time.Second,
//  	Isolation: sql.LevelSerializable,
//  	ReadOnly:  false,
//  }
func ExecuteTransaction(qu Queryer, opt *TxOptions, actions func(*sql.Tx) error) (err error) {
	// Provide safe defaults in case none were given.
	if opt == nil {
		opt = &TxOptions{
			Timeout:   5 * time.Second,
			Isolation: sql.LevelSerializable,
			ReadOnly:  false,
		}
	}

	// Build the context with the provided timeout.
	// This will be used to define the total time the transaction may take,
	// past this time, it will be cancelled, rollback, then throw an error.
	ctx, cancel := context.WithTimeout(context.Background(), opt.Timeout)
	defer cancel()

	var tx *sql.Tx
	if tx, err = qu.BeginTx(ctx, &sql.TxOptions{
		Isolation: opt.Isolation,
		ReadOnly:  opt.ReadOnly,
	}); err != nil {
		return err
	}

	defer func() {
		if r := recover(); r != nil {
			// Only need to log here because panic won't report whether
			// the rollback was successful or not.
			if txerr := tx.Rollback(); txerr != nil {
				log.Println("db rollback error:", txerr)
			}

			log.Printf("rolled back transaction")
			panic(r)
		} else if err != nil {
			// If we run into issues rolling back, keep track of the error that
			// caused the issue and provide some context on the rollback failure.
			if rerr := tx.Rollback(); rerr != nil {
				err = fmt.Errorf("db error: %v rollback error: %v", err, rerr)
			}
		} else {
			if cerr := tx.Commit(); cerr != nil {
				err = fmt.Errorf("commit error: %v", cerr)
			}
		}
	}()

	err = actions(tx)
	return err
}
{{end}}
\"") + packr.PackJSONBytes("./tmpl", "x_helpers_test.html", "\"{{define "helperstest"}}
//+build !helpers

package {{ .PackageName }}

/*---------------------------+
| Code generated by modelgen |
|        DO NOT EDIT.        |
+---------------------------*/

import (
	"database/sql/driver"
	"encoding/json"
	"reflect"
	"testing"
	"time"
)

func TestStructEmbedding(t *testing.T) {
	tim := time.Date(2017, 1, 1, 0, 0, 0, 0, time.UTC)
	expected := []byte(`{"a":123,"b":true,"c":123.123,"d":"string","e":"2017-01-01T00:00:00Z","f":[1,2,3]}`)
	type embed struct {
		A NullInt64   `json:"a,omitempty"`
		B NullBool    `json:"b,omitempty"`
		C NullFloat64 `json:"c,omitempty"`
		D NullString  `json:"d,omitempty"`
		E NullTime    `json:"e,omitempty"`
		F RawJSON     `json:"f,omitempty"`
	}
	em := embed{
		A: NullInt64{Valid: true, Int64: 123},
		B: NullBool{Valid: true, Bool: true},
		C: NullFloat64{Valid: true, Float64: 123.123},
		D: NullString{Valid: true, String: "string"},
		E: NullTime{Valid: true, Time: tim},
		F: RawJSON(`[1,2,3]`),
	}
	b, err := json.Marshal(em)
	if err != nil {
		t.Fatal(err)
	}
	if !reflect.DeepEqual(expected, b) {
		t.Fatal("not the same JSON!")
	}
	if !(string(b) == string(expected)) {
		t.Fatal("not the same!")
	}

	var em2 embed
	if err := json.Unmarshal(expected, &em2); err != nil {
		t.Fatal(err)
	}
	if !reflect.DeepEqual(em2, em) {
		t.Fatal("not correct")
	}
}

func TestNullString_UnmarshalJSON(t *testing.T) {
	tests := []struct {
		name    string
		n       NullString
		source  []byte
		wantErr bool
	}{
		{
			name:    "explicit null",
			source:  []byte(`null`),
			wantErr: false,
		},
		{
			name:    "string null",
			source:  []byte(`"null"`),
			wantErr: false, // this one SHOULD be valid
		},
		{
			name:    "valid",
			source:  []byte(`"hello"`),
			wantErr: false,
		},
		{
			name:    "invalid",
			source:  []byte(`{"key":"value"}`),
			wantErr: true,
		},
		{
			name:    "empty",
			source:  []byte{},
			wantErr: true,
		},
	}
	for _, tt := range tests {
		t.Run(tt.name, func(t *testing.T) {
			if err := tt.n.UnmarshalJSON(tt.source); (err != nil) != tt.wantErr {
				t.Errorf("NullString.UnmarshalJSON() error = %v, wantErr %v", err, tt.wantErr)
			}
		})
	}
}

func TestNullString_Value(t *testing.T) {
	tests := []struct {
		name    string
		n       NullString
		want    driver.Value
		wantErr bool
	}{
		{
			name: "valid",
			n: NullString{
				Valid:  true,
				String: "hello",
			},
			want:    driver.Value("hello"),
			wantErr: false,
		},
		{
			name: "invalid",
			n: NullString{
				Valid: false,
			},
			want:    nil,
			wantErr: false,
		},
	}
	for _, tt := range tests {
		t.Run(tt.name, func(t *testing.T) {
			got, err := tt.n.Value()
			if (err != nil) != tt.wantErr {
				t.Errorf("NullString.Value() error = %v, wantErr %v", err, tt.wantErr)
				return
			}
			if !reflect.DeepEqual(got, tt.want) {
				t.Errorf("NullString.Value() = %v, want %v", got, tt.want)
			}
		})
	}
}

func TestNullString_Scan(t *testing.T) {
	tests := []struct {
		name    string
		n       *NullString
		wantErr bool
		src     interface{}
	}{
		{
			name: "valid",
			n: &NullString{
				String: "hello",
				Valid:  true,
			},
			src:     "",
			wantErr: false,
		},
		{
			name: "nil value",
			n: &NullString{
				String: "hello",
				Valid:  false,
			},
			src:     nil,
			wantErr: false,
		},
	}
	for _, tt := range tests {
		t.Run(tt.name, func(t *testing.T) {
			if err := tt.n.Scan(tt.src); (err != nil) != tt.wantErr {
				t.Errorf("NullString.Scan() error = %v, wantErr %v", err, tt.wantErr)
			}
			if !tt.n.Valid && tt.src != nil {
				t.Errorf("should return null")
			}
			if tt.n.Valid && tt.src != tt.n.String {
				t.Errorf("invalid value")
			}
		})
	}
}

func TestNullString_MarshalJSON(t *testing.T) {
	tests := []struct {
		name    string
		n       *NullString
		want    []byte
		wantErr bool
	}{
		{
			name: "valid",
			n: &NullString{
				String: "hello",
				Valid:  true,
			},
			want:    []byte(`"hello"`),
			wantErr: false,
		},
		{
			name: "valid null",
			n: &NullString{
				String: "",
				Valid:  false,
			},
			want:    []byte(`null`),
			wantErr: false,
		},
		{
			name: "invalid",
			n: &NullString{
				Valid: true,
			},
			want:    []byte(`""`),
			wantErr: false,
		},
	}
	for _, tt := range tests {
		t.Run(tt.name, func(t *testing.T) {
			got, err := tt.n.MarshalJSON()
			if (err != nil) != tt.wantErr {
				t.Errorf("NullString.MarshalJSON() error = %v, wantErr %v", err, tt.wantErr)
				return
			}
			if !reflect.DeepEqual(got, tt.want) {
				t.Errorf("NullString.MarshalJSON() = %v, want %v", got, tt.want)
			}
		})
	}
}

func TestNullBool_UnmarshalJSON(t *testing.T) {
	tests := []struct {
		name         string
		n            NullBool
		source       []byte
		wantErr      bool
		wantValidity bool
	}{
		{
			name:         "explicit null",
			source:       []byte(`null`),
			wantErr:      false,
			wantValidity: false,
		},
		{
			name:         "valid",
			source:       []byte(`false`),
			wantErr:      false,
			wantValidity: true,
		},
		{
			name:         "invalid",
			source:       []byte(`{"key":"value"}`),
			wantErr:      true,
			wantValidity: false,
		},
		{
			name:         "empty",
			source:       []byte{},
			wantErr:      true,
			wantValidity: false,
		},
		{
			name:         "explicit null",
			source:       []byte("null"),
			wantErr:      false,
			wantValidity: false,
		},
	}
	for _, tt := range tests {
		t.Run(tt.name, func(t *testing.T) {
			if err := tt.n.UnmarshalJSON(tt.source); (err != nil) != tt.wantErr && tt.n.Valid == tt.wantValidity {
				t.Errorf("NullBool.UnmarshalJSON() error = %v, wantErr %v", err, tt.wantErr)
			}
		})
	}
}

func TestNullBool_Value(t *testing.T) {
	tests := []struct {
		name    string
		n       NullBool
		want    driver.Value
		wantErr bool
	}{
		{
			name: "valid",
			n: NullBool{
				Valid: true,
				Bool:  true,
			},
			want:    driver.Value(true),
			wantErr: false,
		},
		{
			name: "invalid",
			n: NullBool{
				Valid: false,
			},
			want:    nil,
			wantErr: false,
		},
	}
	for _, tt := range tests {
		t.Run(tt.name, func(t *testing.T) {
			got, err := tt.n.Value()
			if (err != nil) != tt.wantErr {
				t.Errorf("NullBool.Value() error = %v, wantErr %v", err, tt.wantErr)
				return
			}
			if !reflect.DeepEqual(got, tt.want) {
				t.Errorf("NullBool.Value() = %v, want %v", got, tt.want)
			}
		})
	}
}

func TestNullBool_Scan(t *testing.T) {
	tests := []struct {
		name    string
		n       *NullBool
		wantErr bool
		src     interface{}
	}{
		{
			name: "valid",
			n: &NullBool{
				Bool:  true,
				Valid: true,
			},
			src:     true,
			wantErr: false,
		},
		{
			name: "nil value",
			n: &NullBool{
				Bool:  true,
				Valid: false,
			},
			src:     false,
			wantErr: false,
		},
	}
	for _, tt := range tests {
		t.Run(tt.name, func(t *testing.T) {
			if err := tt.n.Scan(tt.src); (err != nil) != tt.wantErr {
				t.Errorf("NullBool.Scan() error = %v, wantErr %v", err, tt.wantErr)
			}
			if !tt.n.Valid && tt.src != nil {
				t.Errorf("should return null")
			}
			if tt.n.Valid && tt.src != tt.n.Bool {
				t.Errorf("invalid value")
			}
		})
	}
}

func TestNullBool_MarshalJSON(t *testing.T) {
	tests := []struct {
		name    string
		n       *NullBool
		want    []byte
		wantErr bool
	}{
		{
			name: "valid",
			n: &NullBool{
				Valid: true,
			},
			want:    []byte(`false`),
			wantErr: false,
		},
		{
			name: "valid null",
			n: &NullBool{
				Valid: false,
			},
			want:    []byte(`null`),
			wantErr: false,
		},
		{
			name: "invalid",
			n: &NullBool{
				Valid: true,
			},
			want:    []byte(`false`),
			wantErr: false,
		},
	}
	for _, tt := range tests {
		t.Run(tt.name, func(t *testing.T) {
			got, err := tt.n.MarshalJSON()
			if (err != nil) != tt.wantErr {
				t.Errorf("NullBool.MarshalJSON() error = %v, wantErr %v", err, tt.wantErr)
				return
			}
			if !reflect.DeepEqual(got, tt.want) {
				t.Errorf("NullBool.MarshalJSON() = %v, want %v", got, tt.want)
			}
		})
	}
}

func TestNullTime_UnmarshalJSON(t *testing.T) {
	tests := []struct {
		name    string
		n       NullTime
		source  []byte
		wantErr bool
	}{
		{
			name:    "valid",
			source:  []byte(`"2017-11-24T00:00:00Z"`),
			wantErr: false,
		},
		{
			name:    "invalid",
			source:  []byte(`{"key":"value"}`),
			wantErr: true,
		},
		{
			name:    "empty",
			source:  []byte{},
			wantErr: true,
		},
		{
			name:    "explicit null",
			source:  []byte(`null`),
			wantErr: false,
		},
	}
	for _, tt := range tests {
		t.Run(tt.name, func(t *testing.T) {
			if err := tt.n.UnmarshalJSON(tt.source); (err != nil) != tt.wantErr {
				t.Errorf("NullBool.UnmarshalJSON() error = %v, wantErr %v", err, tt.wantErr)
			}
		})
	}
}

func TestNullTime_Value(t *testing.T) {
	tim := time.Now()
	tests := []struct {
		name    string
		n       NullTime
		want    driver.Value
		wantErr bool
	}{
		{
			name: "valid",
			n: NullTime{
				Valid: true,
				Time:  tim,
			},
			want:    driver.Value(tim),
			wantErr: false,
		},
		{
			name: "invalid",
			n: NullTime{
				Valid: false,
			},
			want:    nil,
			wantErr: false,
		},
	}
	for _, tt := range tests {
		t.Run(tt.name, func(t *testing.T) {
			got, err := tt.n.Value()
			if (err != nil) != tt.wantErr {
				t.Errorf("NullTime.Value() error = %v, wantErr %v", err, tt.wantErr)
				return
			}
			if !reflect.DeepEqual(got, tt.want) {
				t.Errorf("NullTime.Value() = %v, want %v", got, tt.want)
			}
		})
	}
}

func TestNullTime_Scan(t *testing.T) {
	tim := time.Now()
	tests := []struct {
		name    string
		n       *NullTime
		wantErr bool
		src     interface{}
	}{
		{
			name: "valid",
			n: &NullTime{
				Time:  tim,
				Valid: true,
			},
			src:     tim,
			wantErr: false,
		},
		{
			name: "nil value",
			n: &NullTime{
				Time:  tim,
				Valid: false,
			},
			src:     time.Now(),
			wantErr: false,
		},
	}
	for _, tt := range tests {
		t.Run(tt.name, func(t *testing.T) {
			if err := tt.n.Scan(tt.src); (err != nil) != tt.wantErr {
				t.Errorf("NullTime.Scan() error = %v, wantErr %v", err, tt.wantErr)
			}
			if !tt.n.Valid && tt.src != nil {
				t.Errorf("should return null")
			}
			if tt.n.Valid && tt.src != tt.n.Time {
				t.Errorf("invalid value")
			}
		})
	}
}

func TestNullTime_MarshalJSON(t *testing.T) {
	tests := []struct {
		name    string
		n       *NullTime
		want    []byte
		wantErr bool
	}{
		{
			name: "valid",
			n: &NullTime{
				Time:  time.Date(2017, 11, 24, 0, 0, 0, 0, time.UTC),
				Valid: true,
			},
			want:    []byte(`"2017-11-24T00:00:00Z"`),
			wantErr: false,
		},
		{
			name: "valid null",
			n: &NullTime{
				Valid: false,
			},
			want:    []byte(`null`),
			wantErr: false,
		},
		{
			name: "invalid",
			n: &NullTime{
				Valid: true,
			},
			want:    []byte(`"0001-01-01T00:00:00Z"`),
			wantErr: false,
		},
	}
	for _, tt := range tests {
		t.Run(tt.name, func(t *testing.T) {
			got, err := tt.n.MarshalJSON()
			if (err != nil) != tt.wantErr {
				t.Errorf("NullTime.MarshalJSON() error = %v, wantErr %v", err, tt.wantErr)
				return
			}
			if !reflect.DeepEqual(got, tt.want) {
				t.Errorf("NullTime.MarshalJSON() = %v, want %v", got, tt.want)
			}
		})
	}
}

func TestNullInt64_UnmarshalJSON(t *testing.T) {
	tests := []struct {
		name    string
		n       NullInt64
		source  []byte
		wantErr bool
	}{
		{
			name:    "valid",
			source:  []byte(`123`),
			wantErr: false,
		},
		{
			name:    "invalid",
			source:  []byte(`{"key":"value"}`),
			wantErr: true,
		},
		{
			name:    "empty",
			source:  []byte{},
			wantErr: true,
		},
	}
	for _, tt := range tests {
		t.Run(tt.name, func(t *testing.T) {
			if err := tt.n.UnmarshalJSON(tt.source); (err != nil) != tt.wantErr {
				t.Errorf("NullInt64.UnmarshalJSON() error = %v, wantErr %v", err, tt.wantErr)
			}
		})
	}
}

func TestNullInt64_Value(t *testing.T) {
	tests := []struct {
		name    string
		n       NullInt64
		want    driver.Value
		wantErr bool
	}{
		{
			name: "valid",
			n: NullInt64{
				Valid: true,
				Int64: 123,
			},
			want:    driver.Value(int64(123)),
			wantErr: false,
		},
		{
			name: "invalid",
			n: NullInt64{
				Valid: false,
			},
			want:    nil,
			wantErr: false,
		},
	}
	for _, tt := range tests {
		t.Run(tt.name, func(t *testing.T) {
			got, err := tt.n.Value()
			if (err != nil) != tt.wantErr {
				t.Errorf("NullInt64.Value() error = %v, wantErr %v", err, tt.wantErr)
				return
			}
			if !reflect.DeepEqual(got, tt.want) {
				t.Errorf("NullInt64.Value() = %v, want %v", got, tt.want)
			}
		})
	}
}

func TestNullInt64_Scan(t *testing.T) {
	tests := []struct {
		name    string
		n       *NullInt64
		wantErr bool
		src     interface{}
	}{
		{
			name: "valid",
			n: &NullInt64{
				Int64: 123,
				Valid: true,
			},
			src:     int64(123),
			wantErr: false,
		},
		{
			name: "nil value",
			n: &NullInt64{
				Valid: false,
			},
			src:     int64(123),
			wantErr: false,
		},
	}
	for _, tt := range tests {
		t.Run(tt.name, func(t *testing.T) {
			if err := tt.n.Scan(tt.src); (err != nil) != tt.wantErr {
				t.Errorf("NullInt64.Scan() error = %v, wantErr %v", err, tt.wantErr)
			}
			if !tt.n.Valid && tt.src != nil {
				t.Errorf("should return null")
			}
			if tt.n.Valid && tt.src != tt.n.Int64 {
				t.Errorf("invalid value")
			}
		})
	}
}

func TestNullInt64_MarshalJSON(t *testing.T) {
	tests := []struct {
		name    string
		n       *NullInt64
		want    []byte
		wantErr bool
	}{
		{
			name: "valid",
			n: &NullInt64{
				Int64: 123,
				Valid: true,
			},
			want:    []byte(`123`),
			wantErr: false,
		},
		{
			name: "valid null",
			n: &NullInt64{
				Valid: false,
			},
			want:    []byte(`null`),
			wantErr: false,
		},
		{
			name: "invalid",
			n: &NullInt64{
				Valid: true,
			},
			want:    []byte(`0`),
			wantErr: false,
		},
	}
	for _, tt := range tests {
		t.Run(tt.name, func(t *testing.T) {
			got, err := tt.n.MarshalJSON()
			if (err != nil) != tt.wantErr {
				t.Errorf("NullInt64.MarshalJSON() error = %v, wantErr %v", err, tt.wantErr)
				return
			}
			if !reflect.DeepEqual(got, tt.want) {
				t.Errorf("NullInt64.MarshalJSON() = %v, want %v", got, tt.want)
			}
		})
	}
}
func TestNullFloat64_UnmarshalJSON(t *testing.T) {
	tests := []struct {
		name    string
		n       NullFloat64
		source  []byte
		wantErr bool
	}{
		{
			name:    "explicit null",
			source:  []byte(`null`),
			wantErr: false,
		},
		{
			name:    "valid",
			source:  []byte(`123.123`),
			wantErr: false,
		},
		{
			name:    "invalid",
			source:  []byte(`{"key":"value"}`),
			wantErr: true,
		},
		{
			name:    "empty",
			source:  []byte{},
			wantErr: true,
		},
	}
	for _, tt := range tests {
		t.Run(tt.name, func(t *testing.T) {
			if err := tt.n.UnmarshalJSON(tt.source); (err != nil) != tt.wantErr {
				t.Errorf("NullFloat64.UnmarshalJSON() error = %v, wantErr %v", err, tt.wantErr)
			}
		})
	}
}

func TestNullFloat64_Value(t *testing.T) {
	tests := []struct {
		name    string
		n       NullFloat64
		want    driver.Value
		wantErr bool
	}{
		{
			name: "valid",
			n: NullFloat64{
				Valid:   true,
				Float64: 123.123,
			},
			want:    driver.Value(float64(123.123)),
			wantErr: false,
		},
		{
			name: "invalid",
			n: NullFloat64{
				Valid: false,
			},
			want:    nil,
			wantErr: false,
		},
	}
	for _, tt := range tests {
		t.Run(tt.name, func(t *testing.T) {
			got, err := tt.n.Value()
			if (err != nil) != tt.wantErr {
				t.Errorf("NullFloat64.Value() error = %v, wantErr %v", err, tt.wantErr)
				return
			}
			if !reflect.DeepEqual(got, tt.want) {
				t.Errorf("NullFloat64.Value() = %v, want %v", got, tt.want)
			}
		})
	}
}

func TestNullFloat64_Scan(t *testing.T) {
	tests := []struct {
		name    string
		n       *NullFloat64
		wantErr bool
		src     interface{}
	}{
		{
			name: "valid",
			n: &NullFloat64{
				Float64: 123.123,
				Valid:   true,
			},
			src:     float64(123),
			wantErr: false,
		},
		{
			name: "nil value",
			n: &NullFloat64{
				Valid: false,
			},
			src:     float64(123.123),
			wantErr: false,
		},
	}
	for _, tt := range tests {
		t.Run(tt.name, func(t *testing.T) {
			if err := tt.n.Scan(tt.src); (err != nil) != tt.wantErr {
				t.Errorf("NullFloat64.Scan() error = %v, wantErr %v", err, tt.wantErr)
			}
			if !tt.n.Valid && tt.src != nil {
				t.Errorf("should return null")
			}
			if tt.n.Valid && tt.src != tt.n.Float64 {
				t.Errorf("invalid value")
			}
		})
	}
}

func TestNullFloat64_MarshalJSON(t *testing.T) {
	tests := []struct {
		name    string
		n       *NullFloat64
		want    []byte
		wantErr bool
	}{
		{
			name: "valid",
			n: &NullFloat64{
				Float64: 123.123,
				Valid:   true,
			},
			want:    []byte(`123.123`),
			wantErr: false,
		},
		{
			name: "valid null",
			n: &NullFloat64{
				Valid: false,
			},
			want:    []byte(`null`),
			wantErr: false,
		},
		{
			name: "invalid",
			n: &NullFloat64{
				Valid: true,
			},
			want:    []byte(`0`),
			wantErr: false,
		},
	}
	for _, tt := range tests {
		t.Run(tt.name, func(t *testing.T) {
			got, err := tt.n.MarshalJSON()
			if (err != nil) != tt.wantErr {
				t.Errorf("NullFloat64.MarshalJSON() error = %v, wantErr %v", err, tt.wantErr)
				return
			}
			if !reflect.DeepEqual(got, tt.want) {
				t.Errorf("NullFloat64.MarshalJSON() = %v, want %v", got, tt.want)
			}
		})
	}
}

func TestToNullBool(t *testing.T) {
	b := true
	bb := ToNullBool(&b)
	if !bb.Valid {
		t.Errorf("expected valid, got %v", bb.Valid)
	}
	if !bb.Bool {
		t.Errorf("expected true, got %v", bb.Bool)
	}

	var b2 *bool
	bb2 := ToNullBool(b2)
	if bb2.Valid {
		t.Errorf("expected not valid, got %v", bb2.Valid)
	}
	if bb2.Bool {
		t.Errorf("expected false, got %v", bb2.Bool)
	}
}
func TestToNullInt64(t *testing.T) {
	b := int64(123)
	bb := ToNullInt64(&b)
	if !bb.Valid {
		t.Errorf("expected valid, got %v", bb.Valid)
	}
	if bb.Int64 != 123 {
		t.Errorf("expected 123, got %v", bb.Int64)
	}

	var b2 *int64
	bb2 := ToNullInt64(b2)
	if bb2.Valid {
		t.Errorf("expected not valid, got %v", bb2.Valid)
	}
	if bb2.Int64 != 0 {
		t.Errorf("expected 0, got %v", bb2.Int64)
	}
}

func TestToNullFloat64(t *testing.T) {
	b := float64(123.123)
	bb := ToNullFloat64(&b)
	if !bb.Valid {
		t.Errorf("expected valid, got %v", bb.Valid)
	}
	if bb.Float64 != 123.123 {
		t.Errorf("expected 123.123, got %v", bb.Float64)
	}

	var b2 *float64
	bb2 := ToNullFloat64(b2)
	if bb2.Valid {
		t.Errorf("expected not valid, got %v", bb2.Valid)
	}
	if bb2.Float64 != 0 {
		t.Errorf("expected 0, got %v", bb2.Float64)
	}
}
func TestToNullString(t *testing.T) {
	b := "qwe"
	bb := ToNullString(&b)
	if !bb.Valid {
		t.Errorf("expected valid, got %v", bb.Valid)
	}
	if bb.String != "qwe" {
		t.Errorf("expected qwe, got %v", bb.String)
	}

	var b2 *string
	bb2 := ToNullString(b2)
	if bb2.Valid {
		t.Errorf("expected not valid, got %v", bb2.Valid)
	}
	if bb2.String != "" {
		t.Errorf("expected <empty string>, got %v", bb2.String)
	}
}
func TestToNullTime(t *testing.T) {
	tim := time.Now()
	bb := ToNullTime(tim)
	if !bb.Valid {
		t.Errorf("expected valid, got %v", bb.Valid)
	}
	if bb.Time != tim {
		t.Errorf("expected %v, got %v", tim, bb.Time)
	}

	tim = time.Time{}
	bb = ToNullTime(tim)
	if bb.Valid {
		t.Errorf("expected invalid, got %v", bb.Valid)
	}
	if bb.Time != tim {
		t.Errorf("expected %v, got %v", tim, bb.Time)
	}
}

func TestRawJSON_MarshalJSON(t *testing.T) {
	cases := []struct {
		name string
		data []byte
		exp  string
	}{
		{
			name: "empty data",
			data: []byte{},
			exp:  "null",
		},
	}

	for _, c := range cases {
		t.Run(c.name, func(t *testing.T) {
			rj := RawJSON(c.data)
			b, err := rj.MarshalJSON()
			if err != nil {
				t.Fatalf("unexpected error: %v", err)
			}
			if string(b) != c.exp {
				t.Fatalf("\nexp: %q\ngot: %q", c.exp, string(b))
			}
		})
	}
}
{{end}}
\"") } diff --git a/tmpl/x_helpers.html b/tmpl/x_helpers.html index f83ee14..9274792 100644 --- a/tmpl/x_helpers.html +++ b/tmpl/x_helpers.html @@ -321,6 +321,9 @@ // MarshalJSON for NullString func (n RawJSON) MarshalJSON() ([]byte, error) { + if len(n) == 0 { + return []byte("null"), nil + } a := json.RawMessage(n) return a.MarshalJSON() } diff --git a/tmpl/x_helpers_test.html b/tmpl/x_helpers_test.html index 9eece5b..687060a 100644 --- a/tmpl/x_helpers_test.html +++ b/tmpl/x_helpers_test.html @@ -981,4 +981,31 @@ t.Errorf("expected %v, got %v", tim, bb.Time) } } + +func TestRawJSON_MarshalJSON(t *testing.T) { + cases := []struct { + name string + data []byte + exp string + }{ + { + name: "empty data", + data: []byte{}, + exp: "null", + }, + } + + for _, c := range cases { + t.Run(c.name, func(t *testing.T) { + rj := RawJSON(c.data) + b, err := rj.MarshalJSON() + if err != nil { + t.Fatalf("unexpected error: %v", err) + } + if string(b) != c.exp { + t.Fatalf("\nexp: %q\ngot: %q", c.exp, string(b)) + } + }) + } +} {{end}}