summaryrefslogtreecommitdiff
path: root/tests/corrupt-repo-ref.js
diff options
context:
space:
mode:
authorColin Walters <walters@verbum.org>2014-01-19 12:39:38 -0500
committerColin Walters <walters@verbum.org>2014-01-19 13:19:10 -0500
commit3802a0679b5b0ef7a434c09e2c16b97141fe1cee (patch)
tree77df0d978c414020c8132e3ae845f1ec84e931e1 /tests/corrupt-repo-ref.js
parentf841313206626671300b5a2436a19cd5b92715bf (diff)
downloadostree-3802a0679b5b0ef7a434c09e2c16b97141fe1cee.tar.gz
tests/pull-corruption: Ensure we corrupt an object to be pulled
This test had some nondeterminism because we chose a random object to corrupt, but because there were multiple commits, it was possible that we chose an object that was not being pulled. Fix this by writing some custom GJS code to find an explicitly random object that exists in a given ref, an change a random byte offset. This adds a lot more randomness to the testing too.
Diffstat (limited to 'tests/corrupt-repo-ref.js')
-rw-r--r--tests/corrupt-repo-ref.js88
1 files changed, 88 insertions, 0 deletions
diff --git a/tests/corrupt-repo-ref.js b/tests/corrupt-repo-ref.js
new file mode 100644
index 00000000..b1a797da
--- /dev/null
+++ b/tests/corrupt-repo-ref.js
@@ -0,0 +1,88 @@
+#!/usr/bin/env gjs
+//
+// Copyright (C) 2014 Colin Walters <walters@verbum.org>
+//
+// 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 of the License, or (at your option) any later version.
+//
+// This library is distributed in the hope that it will be useful,
+// but WITHOUT ANY WARRANTY; without even the implied warranty of
+// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+// Lesser General Public License for more details.
+//
+// You should have received a copy of the GNU Lesser General Public
+// License along with this library; if not, write to the
+// Free Software Foundation, Inc., 59 Temple Place - Suite 330,
+// Boston, MA 02111-1307, USA.
+
+const GLib = imports.gi.GLib;
+const Gio = imports.gi.Gio;
+const OSTree = imports.gi.OSTree;
+
+let repoPathArg = ARGV[0];
+let refToCorrupt = ARGV[1];
+
+let repo = OSTree.Repo.new(Gio.File.new_for_path(repoPathArg));
+
+repo.open(null);
+
+function listObjectChecksumsRecurse(dir, allObjects) {
+ dir.ensure_resolved();
+ allObjects[dir.tree_get_contents_checksum() + '.dirtree'] = true;
+ allObjects[dir.get_checksum() + '.dirmeta'] = true;
+ let e = dir.enumerate_children('standard::name,standard::type', 0, null);
+ let info;
+ while ((info = e.next_file(null)) != null) {
+ let child = e.get_child(info);
+ child.ensure_resolved();
+ print(info.get_name() + " is " + info.get_file_type());
+ if (info.get_file_type() == Gio.FileType.DIRECTORY) {
+ listObjectChecksumsRecurse(child, allObjects);
+ } else {
+ allObjects[child.get_checksum() + '.filez'] = true;
+ }
+ }
+ e.close(null);
+}
+
+let [,commit] = repo.resolve_rev(refToCorrupt, false);
+let [,root,commit] = repo.read_commit(refToCorrupt, null);
+let allObjects = {};
+allObjects[commit + '.commit'] = true;
+listObjectChecksumsRecurse(root, allObjects);
+let i = 0;
+for (let v in allObjects)
+ i++;
+print("commit " + commit + " refers to " + i + " objects");
+let offset = GLib.random_int_range(0, i);
+let objectToCorrupt = null;
+for (let v in allObjects) {
+ if (offset <= 0) {
+ objectToCorrupt = v;
+ break;
+ }
+ offset--;
+}
+print("Choosing " + objectToCorrupt + " to corrupt");
+
+let loosePath = repo.get_path().resolve_relative_path('objects/' + objectToCorrupt.substring(0, 2) + "/" + objectToCorrupt.substring(2));
+
+let iostream = loosePath.open_readwrite(null);
+let info = iostream.query_info('standard::size', null);
+let size = info.get_size();
+let byteOffsetToCorrupt = GLib.random_int_range(0, size);
+iostream.seek(byteOffsetToCorrupt, GLib.SeekType.SET, null);
+let datain = Gio.DataInputStream.new(iostream.get_input_stream());
+let dataout = Gio.DataOutputStream.new(iostream.get_output_stream());
+let inbyte = datain.read_byte(null);
+let outbyte = (inbyte + 1) % 255;
+dataout.put_byte(outbyte, null);
+dataout.flush(null);
+iostream.close(null);
+let status = "Changed byte offset " + byteOffsetToCorrupt + " from " + inbyte + " to " + outbyte;
+print(status);
+
+let successFile = Gio.File.new_for_path('corrupted-status.txt');
+successFile.replace_contents(status, null, false, 0, null);