You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

DistributedAnswer.java 3.1KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293
  1. /*
  2. * SonarQube
  3. * Copyright (C) 2009-2021 SonarSource SA
  4. * mailto:info AT sonarsource DOT com
  5. *
  6. * This program is free software; you can redistribute it and/or
  7. * modify it under the terms of the GNU Lesser General Public
  8. * License as published by the Free Software Foundation; either
  9. * version 3 of the License, or (at your option) any later version.
  10. *
  11. * This program is distributed in the hope that it will be useful,
  12. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  13. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
  14. * Lesser General Public License for more details.
  15. *
  16. * You should have received a copy of the GNU Lesser General Public License
  17. * along with this program; if not, write to the Free Software Foundation,
  18. * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
  19. */
  20. package org.sonar.process.cluster.hz;
  21. import com.hazelcast.cluster.Member;
  22. import java.util.ArrayList;
  23. import java.util.Collection;
  24. import java.util.HashMap;
  25. import java.util.HashSet;
  26. import java.util.List;
  27. import java.util.Map;
  28. import java.util.Optional;
  29. import java.util.Set;
  30. import java.util.stream.Collectors;
  31. import static org.sonar.process.cluster.hz.HazelcastMember.Attribute.NODE_NAME;
  32. /**
  33. * Answer of {@link DistributedCall}, aggregating the answers from
  34. * all the target members.
  35. */
  36. public class DistributedAnswer<T> {
  37. private final Map<Member, T> answers = new HashMap<>();
  38. private final Set<Member> timedOutMembers = new HashSet<>();
  39. private final Map<Member, Exception> failedMembers = new HashMap<>();
  40. public Optional<T> getAnswer(Member member) {
  41. return Optional.ofNullable(answers.get(member));
  42. }
  43. public boolean hasTimedOut(Member member) {
  44. return timedOutMembers.contains(member);
  45. }
  46. public Optional<Exception> getFailed(Member member) {
  47. return Optional.ofNullable(failedMembers.get(member));
  48. }
  49. public Collection<Member> getMembers() {
  50. List<Member> members = new ArrayList<>();
  51. members.addAll(answers.keySet());
  52. members.addAll(timedOutMembers);
  53. members.addAll(failedMembers.keySet());
  54. return members;
  55. }
  56. public void setAnswer(Member member, T answer) {
  57. this.answers.put(member, answer);
  58. }
  59. public void setTimedOut(Member member) {
  60. this.timedOutMembers.add(member);
  61. }
  62. public void setFailed(Member member, Exception e) {
  63. failedMembers.put(member, e);
  64. }
  65. public void propagateExceptions() {
  66. if (!failedMembers.isEmpty()) {
  67. String failedMemberNames = failedMembers.keySet().stream()
  68. .map(m -> m.getAttribute(NODE_NAME.getKey()))
  69. .collect(Collectors.joining(", "));
  70. throw new IllegalStateException("Distributed cluster action in cluster nodes " + failedMemberNames + " (other nodes may have timed out)",
  71. failedMembers.values().iterator().next());
  72. }
  73. if (!timedOutMembers.isEmpty()) {
  74. String timedOutMemberNames = timedOutMembers.stream()
  75. .map(m -> m.getAttribute(NODE_NAME.getKey()))
  76. .collect(Collectors.joining(", "));
  77. throw new IllegalStateException("Distributed cluster action timed out in cluster nodes " + timedOutMemberNames);
  78. }
  79. }
  80. }