a3470f
From 6fb1804d113ae996e085ef0f23fa8908d167f006 Mon Sep 17 00:00:00 2001
a3470f
From: Ravishankar N <ravishankar@redhat.com>
a3470f
Date: Thu, 14 Jun 2018 12:59:06 +0530
a3470f
Subject: [PATCH 299/305] afr: heal gfids when file is not present on all
a3470f
 bricks
a3470f
a3470f
Backport of: https://review.gluster.org/#/c/20271/
a3470f
commit f73814ad08d552d94d0139b2592175d206e7a166 (rhgs) introduced a regression
a3470f
wherein if a file is present in only 1 brick of replica *and* doesn't
a3470f
have a gfid associated with it, it doesn't get healed upon the next
a3470f
lookup from the client. Fix it.
a3470f
a3470f
Change-Id: I7d1111dcb45b1b8b8340a7d02558f05df70aa599
a3470f
BUG: 1592666
a3470f
Signed-off-by: Ravishankar N <ravishankar@redhat.com>
a3470f
Reviewed-on: https://code.engineering.redhat.com/gerrit/141899
a3470f
Tested-by: RHGS Build Bot <nigelb@redhat.com>
a3470f
Reviewed-by: Karthik Subrahmanya <ksubrahm@redhat.com>
a3470f
---
a3470f
 .../replicate/bug-1591193-assign-gfid-and-heal.t   | 128 +++++++++++++++++++++
a3470f
 xlators/cluster/afr/src/afr-self-heal-common.c     |  39 ++++++-
a3470f
 xlators/cluster/afr/src/afr-self-heal-data.c       |   8 +-
a3470f
 xlators/cluster/afr/src/afr-self-heal-entry.c      |   4 +-
a3470f
 xlators/cluster/afr/src/afr-self-heal-name.c       |   6 +-
a3470f
 xlators/cluster/afr/src/afr-self-heal.h            |   6 +-
a3470f
 6 files changed, 179 insertions(+), 12 deletions(-)
a3470f
 create mode 100644 tests/bugs/replicate/bug-1591193-assign-gfid-and-heal.t
a3470f
a3470f
diff --git a/tests/bugs/replicate/bug-1591193-assign-gfid-and-heal.t b/tests/bugs/replicate/bug-1591193-assign-gfid-and-heal.t
a3470f
new file mode 100644
a3470f
index 0000000..d3b5f9a
a3470f
--- /dev/null
a3470f
+++ b/tests/bugs/replicate/bug-1591193-assign-gfid-and-heal.t
a3470f
@@ -0,0 +1,128 @@
a3470f
+#!/bin/bash
a3470f
+
a3470f
+. $(dirname $0)/../../include.rc
a3470f
+. $(dirname $0)/../../volume.rc
a3470f
+. $(dirname $0)/../../afr.rc
a3470f
+
a3470f
+cleanup;
a3470f
+
a3470f
+function check_gfid_and_link_count
a3470f
+{
a3470f
+        local file=$1
a3470f
+
a3470f
+        file_gfid_b0=$(gf_get_gfid_xattr $B0/${V0}0/$file)
a3470f
+        TEST [ ! -z $file_gfid_b0 ]
a3470f
+        file_gfid_b1=$(gf_get_gfid_xattr $B0/${V0}1/$file)
a3470f
+        file_gfid_b2=$(gf_get_gfid_xattr $B0/${V0}2/$file)
a3470f
+        EXPECT $file_gfid_b0 echo $file_gfid_b1
a3470f
+        EXPECT $file_gfid_b0 echo $file_gfid_b2
a3470f
+
a3470f
+        EXPECT "2" stat -c %h $B0/${V0}0/$file
a3470f
+        EXPECT "2" stat -c %h $B0/${V0}1/$file
a3470f
+        EXPECT "2" stat -c %h $B0/${V0}2/$file
a3470f
+}
a3470f
+TESTS_EXPECTED_IN_LOOP=30
a3470f
+
a3470f
+##############################################################################
a3470f
+# Test on 1x3 volume
a3470f
+TEST glusterd
a3470f
+TEST pidof glusterd
a3470f
+TEST $CLI volume create $V0 replica 3 $H0:$B0/${V0}{0,1,2};
a3470f
+TEST $CLI volume start $V0;
a3470f
+
a3470f
+EXPECT_WITHIN $PROCESS_UP_TIMEOUT "1" brick_up_status $V0 $H0 $B0/${V0}0
a3470f
+EXPECT_WITHIN $PROCESS_UP_TIMEOUT "1" brick_up_status $V0 $H0 $B0/${V0}1
a3470f
+EXPECT_WITHIN $PROCESS_UP_TIMEOUT "1" brick_up_status $V0 $H0 $B0/${V0}2
a3470f
+
a3470f
+EXPECT_WITHIN $PROCESS_UP_TIMEOUT "Y" glustershd_up_status
a3470f
+EXPECT_WITHIN $CHILD_UP_TIMEOUT "1" afr_child_up_status_in_shd $V0 0
a3470f
+EXPECT_WITHIN $CHILD_UP_TIMEOUT "1" afr_child_up_status_in_shd $V0 1
a3470f
+EXPECT_WITHIN $CHILD_UP_TIMEOUT "1" afr_child_up_status_in_shd $V0 2
a3470f
+
a3470f
+TEST glusterfs --volfile-id=/$V0 --volfile-server=$H0 --attribute-timeout=0 --entry-timeout=0 $M0
a3470f
+EXPECT_WITHIN $PROCESS_UP_TIMEOUT "1" afr_child_up_status $V0 0
a3470f
+EXPECT_WITHIN $PROCESS_UP_TIMEOUT "1" afr_child_up_status $V0 1
a3470f
+EXPECT_WITHIN $PROCESS_UP_TIMEOUT "1" afr_child_up_status $V0 2
a3470f
+
a3470f
+
a3470f
+# Create files directly in the backend on different bricks
a3470f
+echo $RANDOM >> $B0/${V0}0/file1
a3470f
+echo $RANDOM >> $B0/${V0}1/file2
a3470f
+echo $RANDOM >> $B0/${V0}2/file3
a3470f
+
a3470f
+# To prevent is_fresh_file code path
a3470f
+sleep 2
a3470f
+
a3470f
+# Access them from mount to trigger name + gfid heal.
a3470f
+TEST stat $M0/file1
a3470f
+TEST stat $M0/file2
a3470f
+TEST stat $M0/file3
a3470f
+
a3470f
+# Launch index heal to complete any pending data/metadata heals.
a3470f
+TEST $CLI volume heal $V0
a3470f
+EXPECT_WITHIN $HEAL_TIMEOUT "^0$" get_pending_heal_count $V0
a3470f
+
a3470f
+# Check each file has a gfid and the .glusterfs hardlink
a3470f
+check_gfid_and_link_count file1
a3470f
+check_gfid_and_link_count file2
a3470f
+check_gfid_and_link_count file3
a3470f
+
a3470f
+TEST rm $M0/file1
a3470f
+TEST rm $M0/file2
a3470f
+TEST rm $M0/file3
a3470f
+cleanup;
a3470f
+
a3470f
+##############################################################################
a3470f
+# Test on 1x (2+1) volume
a3470f
+TEST glusterd
a3470f
+TEST pidof glusterd
a3470f
+TEST $CLI volume create $V0 replica 3 arbiter 1 $H0:$B0/${V0}{0,1,2};
a3470f
+TEST $CLI volume start $V0;
a3470f
+
a3470f
+EXPECT_WITHIN $PROCESS_UP_TIMEOUT "1" brick_up_status $V0 $H0 $B0/${V0}0
a3470f
+EXPECT_WITHIN $PROCESS_UP_TIMEOUT "1" brick_up_status $V0 $H0 $B0/${V0}1
a3470f
+EXPECT_WITHIN $PROCESS_UP_TIMEOUT "1" brick_up_status $V0 $H0 $B0/${V0}2
a3470f
+
a3470f
+EXPECT_WITHIN $PROCESS_UP_TIMEOUT "Y" glustershd_up_status
a3470f
+EXPECT_WITHIN $CHILD_UP_TIMEOUT "1" afr_child_up_status_in_shd $V0 0
a3470f
+EXPECT_WITHIN $CHILD_UP_TIMEOUT "1" afr_child_up_status_in_shd $V0 1
a3470f
+EXPECT_WITHIN $CHILD_UP_TIMEOUT "1" afr_child_up_status_in_shd $V0 2
a3470f
+
a3470f
+TEST glusterfs --volfile-id=/$V0 --volfile-server=$H0 --attribute-timeout=0 --entry-timeout=0 $M0
a3470f
+EXPECT_WITHIN $PROCESS_UP_TIMEOUT "1" afr_child_up_status $V0 0
a3470f
+EXPECT_WITHIN $PROCESS_UP_TIMEOUT "1" afr_child_up_status $V0 1
a3470f
+EXPECT_WITHIN $PROCESS_UP_TIMEOUT "1" afr_child_up_status $V0 2
a3470f
+
a3470f
+
a3470f
+# Create files directly in the backend on different bricks
a3470f
+echo $RANDOM >> $B0/${V0}0/file1
a3470f
+echo $RANDOM >> $B0/${V0}1/file2
a3470f
+touch $B0/${V0}2/file3
a3470f
+
a3470f
+# To prevent is_fresh_file code path
a3470f
+sleep 2
a3470f
+
a3470f
+# Access them from mount to trigger name + gfid heal.
a3470f
+TEST stat $M0/file1
a3470f
+TEST stat $M0/file2
a3470f
+
a3470f
+# Though file is created on all 3 bricks, lookup will fail as arbiter blames the
a3470f
+# other 2 bricks and ariter is not 'readable'.
a3470f
+# TEST ! stat $M0/file3
a3470f
+# But the checks for failing lookups when quorum is not met is not yet there in
a3470f
+# rhgs-3.4.0, so stat will succeed.
a3470f
+TEST  stat $M0/file3
a3470f
+
a3470f
+# Launch index heal to complete any pending data/metadata heals.
a3470f
+TEST $CLI volume heal $V0
a3470f
+EXPECT_WITHIN $HEAL_TIMEOUT "^0$" get_pending_heal_count $V0
a3470f
+
a3470f
+# Check each file has a gfid and the .glusterfs hardlink
a3470f
+check_gfid_and_link_count file1
a3470f
+check_gfid_and_link_count file2
a3470f
+check_gfid_and_link_count file3
a3470f
+
a3470f
+TEST rm $M0/file1
a3470f
+TEST rm $M0/file2
a3470f
+TEST rm $M0/file3
a3470f
+cleanup;
a3470f
diff --git a/xlators/cluster/afr/src/afr-self-heal-common.c b/xlators/cluster/afr/src/afr-self-heal-common.c
a3470f
index 32fd24a..50989d6 100644
a3470f
--- a/xlators/cluster/afr/src/afr-self-heal-common.c
a3470f
+++ b/xlators/cluster/afr/src/afr-self-heal-common.c
a3470f
@@ -22,7 +22,7 @@ afr_heal_synctask (xlator_t *this, afr_local_t *local);
a3470f
 int
a3470f
 afr_lookup_and_heal_gfid (xlator_t *this, inode_t *parent, const char *name,
a3470f
                           inode_t *inode, struct afr_reply *replies,
a3470f
-                          int source,  void *gfid)
a3470f
+                          int source,  unsigned char *sources, void *gfid)
a3470f
 {
a3470f
         afr_private_t *priv = NULL;
a3470f
         call_frame_t   *frame    = NULL;
a3470f
@@ -37,6 +37,23 @@ afr_lookup_and_heal_gfid (xlator_t *this, inode_t *parent, const char *name,
a3470f
         priv = this->private;
a3470f
         wind_on = alloca0 (priv->child_count);
a3470f
         ia_type = replies[source].poststat.ia_type;
a3470f
+        if ((ia_type == IA_INVAL) &&
a3470f
+            (AFR_COUNT(sources, priv->child_count) == priv->child_count)) {
a3470f
+                /* If a file is present on some bricks of the replica but parent
a3470f
+                 * dir does not have pending xattrs, all bricks are sources and
a3470f
+                 * the 'source' we  selected earlier might be one where the file
a3470f
+                 * is not actually present. Hence check if file is present in
a3470f
+                 * any of the sources.*/
a3470f
+                for (i = 0; i < priv->child_count; i++) {
a3470f
+                        if (i == source)
a3470f
+                                continue;
a3470f
+                        if (sources[i] && replies[i].valid &&
a3470f
+                            replies[i].op_ret == 0) {
a3470f
+                                ia_type = replies[i].poststat.ia_type;
a3470f
+                                break;
a3470f
+                        }
a3470f
+                }
a3470f
+        }
a3470f
 
a3470f
         /* gfid heal on those subvolumes that do not have gfid associated
a3470f
          * with the inode and update those replies.
a3470f
@@ -1250,6 +1267,21 @@ afr_mark_split_brain_source_sinks_by_policy (call_frame_t *frame,
a3470f
         return fav_child;
a3470f
 }
a3470f
 
a3470f
+gf_boolean_t
a3470f
+afr_is_file_empty_on_all_children (afr_private_t *priv,
a3470f
+                                   struct afr_reply *replies)
a3470f
+{
a3470f
+        int i = 0;
a3470f
+
a3470f
+        for (i = 0; i < priv->child_count; i++) {
a3470f
+                if ((!replies[i].valid) || (replies[i].op_ret != 0) ||
a3470f
+                    (replies[i].poststat.ia_size != 0))
a3470f
+                        return _gf_false;
a3470f
+        }
a3470f
+
a3470f
+        return _gf_true;
a3470f
+}
a3470f
+
a3470f
 int
a3470f
 afr_mark_source_sinks_if_file_empty (xlator_t *this, unsigned char *sources,
a3470f
                                      unsigned char *sinks,
a3470f
@@ -1268,11 +1300,8 @@ afr_mark_source_sinks_if_file_empty (xlator_t *this, unsigned char *sources,
a3470f
                 return -1;
a3470f
 
a3470f
         if (type == AFR_DATA_TRANSACTION) {
a3470f
-                for (i = 0; i < priv->child_count; i++) {
a3470f
-                        if (replies[i].poststat.ia_size != 0)
a3470f
+                if (!afr_is_file_empty_on_all_children(priv, replies))
a3470f
                                 return -1;
a3470f
-                }
a3470f
-
a3470f
                 goto mark;
a3470f
         }
a3470f
 
a3470f
diff --git a/xlators/cluster/afr/src/afr-self-heal-data.c b/xlators/cluster/afr/src/afr-self-heal-data.c
a3470f
index f872a98..3ef7376 100644
a3470f
--- a/xlators/cluster/afr/src/afr-self-heal-data.c
a3470f
+++ b/xlators/cluster/afr/src/afr-self-heal-data.c
a3470f
@@ -670,6 +670,7 @@ __afr_selfheal_data (call_frame_t *frame, xlator_t *this, fd_t *fd,
a3470f
 	int source = -1;
a3470f
         gf_boolean_t did_sh = _gf_true;
a3470f
         gf_boolean_t is_arbiter_the_only_sink = _gf_false;
a3470f
+        gf_boolean_t empty_file = _gf_false;
a3470f
 
a3470f
 	priv = this->private;
a3470f
 
a3470f
@@ -710,6 +711,11 @@ __afr_selfheal_data (call_frame_t *frame, xlator_t *this, fd_t *fd,
a3470f
 		source = ret;
a3470f
 
a3470f
                 if (AFR_IS_ARBITER_BRICK(priv, source)) {
a3470f
+                        empty_file = afr_is_file_empty_on_all_children (priv,
a3470f
+                                                                locked_replies);
a3470f
+                        if (empty_file)
a3470f
+                                goto restore_time;
a3470f
+
a3470f
                         did_sh = _gf_false;
a3470f
                         goto unlock;
a3470f
                 }
a3470f
@@ -746,7 +752,7 @@ restore_time:
a3470f
 	afr_selfheal_restore_time (frame, this, fd->inode, source,
a3470f
 					healed_sinks, locked_replies);
a3470f
 
a3470f
-        if (!is_arbiter_the_only_sink) {
a3470f
+        if (!is_arbiter_the_only_sink || !empty_file) {
a3470f
                 ret = afr_selfheal_inodelk (frame, this, fd->inode, this->name,
a3470f
                                             0, 0, data_lock);
a3470f
                 if (ret < priv->child_count) {
a3470f
diff --git a/xlators/cluster/afr/src/afr-self-heal-entry.c b/xlators/cluster/afr/src/afr-self-heal-entry.c
a3470f
index 647dd71..f6d3a8a 100644
a3470f
--- a/xlators/cluster/afr/src/afr-self-heal-entry.c
a3470f
+++ b/xlators/cluster/afr/src/afr-self-heal-entry.c
a3470f
@@ -187,7 +187,7 @@ __afr_selfheal_heal_dirent (call_frame_t *frame, xlator_t *this, fd_t *fd,
a3470f
 
a3470f
         if (replies[source].op_ret == 0) {
a3470f
                 ret = afr_lookup_and_heal_gfid (this, fd->inode, name,
a3470f
-                                                inode, replies, source,
a3470f
+                                                inode, replies, source, sources,
a3470f
                                              &replies[source].poststat.ia_gfid);
a3470f
                 if (ret)
a3470f
                         return ret;
a3470f
@@ -320,7 +320,7 @@ __afr_selfheal_merge_dirent (call_frame_t *frame, xlator_t *this, fd_t *fd,
a3470f
 	}
a3470f
 
a3470f
         ret = afr_lookup_and_heal_gfid (this, fd->inode, name, inode, replies,
a3470f
-                                        source,
a3470f
+                                        source, sources,
a3470f
                                         &replies[source].poststat.ia_gfid);
a3470f
         if (ret)
a3470f
                 return ret;
a3470f
diff --git a/xlators/cluster/afr/src/afr-self-heal-name.c b/xlators/cluster/afr/src/afr-self-heal-name.c
a3470f
index 556d14b..bcd0e60 100644
a3470f
--- a/xlators/cluster/afr/src/afr-self-heal-name.c
a3470f
+++ b/xlators/cluster/afr/src/afr-self-heal-name.c
a3470f
@@ -19,7 +19,7 @@ __afr_selfheal_assign_gfid (xlator_t *this, inode_t *parent, uuid_t pargfid,
a3470f
                             const char *bname, inode_t *inode,
a3470f
                             struct afr_reply *replies, void *gfid,
a3470f
                             unsigned char *locked_on, int source,
a3470f
-                            gf_boolean_t is_gfid_absent)
a3470f
+                            unsigned char *sources, gf_boolean_t is_gfid_absent)
a3470f
 {
a3470f
 	int             ret          = 0;
a3470f
         int             up_count     = 0;
a3470f
@@ -48,7 +48,7 @@ __afr_selfheal_assign_gfid (xlator_t *this, inode_t *parent, uuid_t pargfid,
a3470f
         }
a3470f
 
a3470f
         afr_lookup_and_heal_gfid (this, parent, bname, inode, replies, source,
a3470f
-                                  gfid);
a3470f
+                                  sources, gfid);
a3470f
 
a3470f
 out:
a3470f
 	return ret;
a3470f
@@ -426,7 +426,7 @@ __afr_selfheal_name_do (call_frame_t *frame, xlator_t *this, inode_t *parent,
a3470f
         is_gfid_absent = (gfid_idx == -1) ? _gf_true : _gf_false;
a3470f
 	ret = __afr_selfheal_assign_gfid (this, parent, pargfid, bname, inode,
a3470f
                                           replies, gfid, locked_on, source,
a3470f
-                                          is_gfid_absent);
a3470f
+                                          sources, is_gfid_absent);
a3470f
         if (ret)
a3470f
                 return ret;
a3470f
 
a3470f
diff --git a/xlators/cluster/afr/src/afr-self-heal.h b/xlators/cluster/afr/src/afr-self-heal.h
a3470f
index b015976..cc99d9e 100644
a3470f
--- a/xlators/cluster/afr/src/afr-self-heal.h
a3470f
+++ b/xlators/cluster/afr/src/afr-self-heal.h
a3470f
@@ -113,7 +113,7 @@ afr_selfheal_entry (call_frame_t *frame, xlator_t *this, inode_t *inode);
a3470f
 int
a3470f
 afr_lookup_and_heal_gfid (xlator_t *this, inode_t *parent, const char *name,
a3470f
                           inode_t *inode, struct afr_reply *replies, int source,
a3470f
-                          void *gfid);
a3470f
+                          unsigned char *sources, void *gfid);
a3470f
 
a3470f
 int
a3470f
 afr_selfheal_inodelk (call_frame_t *frame, xlator_t *this, inode_t *inode,
a3470f
@@ -354,4 +354,8 @@ afr_mark_source_sinks_if_file_empty (xlator_t *this, unsigned char *sources,
a3470f
                                      struct afr_reply *replies,
a3470f
                                      afr_transaction_type type);
a3470f
 
a3470f
+gf_boolean_t
a3470f
+afr_is_file_empty_on_all_children (afr_private_t *priv,
a3470f
+                                   struct afr_reply *replies);
a3470f
+
a3470f
 #endif /* !_AFR_SELFHEAL_H */
a3470f
-- 
a3470f
1.8.3.1
a3470f