1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
|
// -*- mode:C++; tab-width:8; c-basic-offset:2; indent-tabs-mode:t -*-
// vim: ts=8 sw=2 smarttab
/*
* Ceph - scalable distributed file system
*
* Copyright (C) 2018 Red Hat Inc.
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2.1 of the License, or (at your option) any later version.
*
*/
#include <errno.h>
#include <gtest/gtest.h>
#include "common/ceph_json.h"
#include "common/Clock.h"
#include "common/StackStringStream.h"
#include <sstream>
using namespace std;
TEST(formatter, bug_37706) {
vector<std::string> pgs;
string outstring =
"{\"pg_ready\":true, \"pg_stats\":[ { \"pgid\":\"1.0\", \"version\":\"16'56\",\"reported_seq\":\"62\",\"reported_epoch\":\"20\",\"state\":\"active+clean+inconsistent\",\"last_fresh\":\"2018-12-18 15:21:22.173804\",\"last_change\":\"2018-12-18 15:21:22.173804\",\"last_active\":\"2018-12-18 15:21:22.173804\",\"last_peered\":\"2018-12-18 15:21:22.173804\",\"last_clean\":\"2018-12-18 15:21:22.173804\",\"last_became_active\":\"2018-12-18 15:21:21.347685\",\"last_became_peered\":\"2018-12-18 15:21:21.347685\",\"last_unstale\":\"2018-12-18 15:21:22.173804\",\"last_undegraded\":\"2018-12-18 15:21:22.173804\",\"last_fullsized\":\"2018-12-18 15:21:22.173804\",\"mapping_epoch\":19,\"log_start\":\"0'0\",\"ondisk_log_start\":\"0'0\",\"created\":7,\"last_epoch_clean\":20,\"parent\":\"0.0\",\"parent_split_bits\":0,\"last_scrub\":\"16'56\",\"last_scrub_stamp\":\"2018-12-18 15:21:22.173684\",\"last_deep_scrub\":\"0'0\",\"last_deep_scrub_stamp\":\"2018-12-18 15:21:06.514438\",\"last_clean_scrub_stamp\":\"2018-12-18 15:21:06.514438\",\"log_size\":56,\"ondisk_log_size\":56,\"stats_invalid\":false,\"dirty_stats_invalid\":false,\"omap_stats_invalid\":false,\"hitset_stats_invalid\":false,\"hitset_bytes_stats_invalid\":false,\"pin_stats_invalid\":false,\"manifest_stats_invalid\":false,\"snaptrimq_len\":0,\"stat_sum\":{\"num_bytes\":24448,\"num_objects\":36,\"num_object_clones\":20,\"num_object_copies\":36,\"num_objects_missing_on_primary\":0,\"num_objects_missing\":0,\"num_objects_degraded\":0,\"num_objects_misplaced\":0,\"num_objects_unfound\":0,\"num_objects_dirty\":36,\"num_whiteouts\":3,\"num_read\":0,\"num_read_kb\":0,\"num_write\":36,\"num_write_kb\":50,\"num_scrub_errors\":20,\"num_shallow_scrub_errors\":20,\"num_deep_scrub_errors\":0,\"num_objects_recovered\":0,\"num_bytes_recovered\":0,\"num_keys_recovered\":0,\"num_objects_omap\":0,\"num_objects_hit_set_archive\":0,\"num_bytes_hit_set_archive\":0,\"num_flush\":0,\"num_flush_kb\":0,\"num_evict\":0,\"num_evict_kb\":0,\"num_promote\":0,\"num_flush_mode_high\":0,\"num_flush_mode_low\":0,\"num_evict_mode_some\":0,\"num_evict_mode_full\":0,\"num_objects_pinned\":0,\"num_legacy_snapsets\":0,\"num_large_omap_objects\":0,\"num_objects_manifest\":0},\"up\":[0],\"acting\":[0],\"blocked_by\":[],\"up_primary\":0,\"acting_primary\":0,\"purged_snaps\":[] }]}";
JSONParser parser;
ASSERT_TRUE(parser.parse(outstring.c_str(), outstring.size()));
vector<string> v;
ASSERT_TRUE (!parser.is_array());
JSONObj *pgstat_obj = parser.find_obj("pg_stats");
ASSERT_TRUE (!!pgstat_obj);
auto s = pgstat_obj->get_data();
ASSERT_TRUE(!s.empty());
JSONParser pg_stats;
ASSERT_TRUE(pg_stats.parse(s.c_str(), s.length()));
v = pg_stats.get_array_elements();
for (auto i : v) {
JSONParser pg_json;
ASSERT_TRUE(pg_json.parse(i.c_str(), i.length()));
string pgid;
JSONDecoder::decode_json("pgid", pgid, &pg_json);
pgs.emplace_back(std::move(pgid));
}
ASSERT_EQ(pgs.back(), "1.0");
}
TEST(formatter, utime)
{
JSONFormatter formatter;
utime_t input = ceph_clock_now();
input.gmtime_nsec(formatter.dump_stream("timestamp"));
bufferlist bl;
formatter.flush(bl);
JSONParser parser;
EXPECT_TRUE(parser.parse(bl.c_str(), bl.length()));
cout << input << " -> '" << std::string(bl.c_str(), bl.length())
<< std::endl;
utime_t output;
decode_json_obj(output, &parser);
cout << " -> " << output << std::endl;
EXPECT_EQ(input.sec(), output.sec());
EXPECT_EQ(input.nsec(), output.nsec());
}
TEST(formatter, dump_inf_or_nan)
{
JSONFormatter formatter;
formatter.open_object_section("inf_and_nan");
double inf = std::numeric_limits<double>::infinity();
formatter.dump_float("positive_infinity", inf);
formatter.dump_float("negative_infinity", -inf);
formatter.dump_float("nan_val", std::numeric_limits<double>::quiet_NaN());
formatter.dump_float("nan_val_alt", std::nan(""));
formatter.close_section();
bufferlist bl;
formatter.flush(bl);
std::cout << std::string(bl.c_str(), bl.length()) << std::endl;
JSONParser parser;
parser.parse(bl.c_str(), bl.length());
EXPECT_TRUE(parser.parse(bl.c_str(), bl.length()));
EXPECT_EQ(parser.find_obj("positive_infinity")->get_data(), "null");
EXPECT_EQ(parser.find_obj("negative_infinity")->get_data(), "null");
EXPECT_EQ(parser.find_obj("nan_val")->get_data(), "null");
EXPECT_EQ(parser.find_obj("nan_val_alt")->get_data(), "null");
}
TEST(formatter, dump_large_item) {
JSONFormatter formatter;
formatter.open_object_section("large_item");
std::string base_url("http://example.com");
std::string bucket_name("bucket");
std::string object_key(1024, 'a');
std::string full_url = base_url + "/" + bucket_name + "/" + object_key;
formatter.dump_format("Location", "%s/%s/%s", base_url.c_str(), bucket_name.c_str(), object_key.c_str());
formatter.close_section();
bufferlist bl;
formatter.flush(bl);
// std::cout << std::string(bl.c_str(), bl.length()) << std::endl;
JSONParser parser;
parser.parse(bl.c_str(), bl.length());
EXPECT_TRUE(parser.parse(bl.c_str(), bl.length()));
EXPECT_EQ(parser.find_obj("Location")->get_data(), full_url);
}
|