[127] | 1 | package negotiator.parties;
|
---|
| 2 |
|
---|
| 3 | import static java.lang.Math.E;
|
---|
| 4 | import static java.lang.Math.max;
|
---|
| 5 | import static java.lang.Math.pow;
|
---|
| 6 |
|
---|
| 7 | import java.util.List;
|
---|
| 8 |
|
---|
| 9 | import genius.core.AgentID;
|
---|
| 10 | import genius.core.DeadlineType;
|
---|
| 11 | import genius.core.Vote;
|
---|
| 12 | import genius.core.actions.Action;
|
---|
| 13 | import genius.core.actions.InformVotingResult;
|
---|
| 14 | import genius.core.actions.OfferForVoting;
|
---|
| 15 | import genius.core.actions.VoteForOfferAcceptance;
|
---|
| 16 | import genius.core.parties.AbstractNegotiationParty;
|
---|
| 17 | import genius.core.parties.NegotiationInfo;
|
---|
| 18 | import genius.core.parties.NegotiationParty;
|
---|
| 19 | import genius.core.protocol.MultilateralProtocol;
|
---|
| 20 |
|
---|
| 21 | /**
|
---|
| 22 | * Implementation of a party that uses simulated annealing strategy to get to an
|
---|
| 23 | * agreement.
|
---|
| 24 | * <p/>
|
---|
| 25 | * This party should be run with {@link genius.core.protocol.MediatorProtocol}
|
---|
| 26 | *
|
---|
| 27 | * @author David Festen
|
---|
| 28 | * @author Reyhan
|
---|
| 29 | */
|
---|
| 30 | public class Annealer extends AbstractNegotiationParty {
|
---|
| 31 |
|
---|
| 32 | /**
|
---|
| 33 | * Holds the utility value for the most recently accepted offer.
|
---|
| 34 | */
|
---|
| 35 | private double lastAcceptedBidUtility;
|
---|
| 36 |
|
---|
| 37 | /**
|
---|
| 38 | * Holds the utility value for the most recently received offer.
|
---|
| 39 | */
|
---|
| 40 | private double lastReceivedBidUtility;
|
---|
| 41 |
|
---|
| 42 | /**
|
---|
| 43 | * Holds the vote that will be done when asked for voting.
|
---|
| 44 | */
|
---|
| 45 | private Vote currentVote;
|
---|
| 46 |
|
---|
| 47 | /**
|
---|
| 48 | * Holds the current round number (used for cooling down the annealing).
|
---|
| 49 | */
|
---|
| 50 | private int currentRound;
|
---|
| 51 |
|
---|
| 52 | @Override
|
---|
| 53 | public void init(NegotiationInfo info) {
|
---|
| 54 | super.init(info);
|
---|
| 55 | lastAcceptedBidUtility = 0;
|
---|
| 56 | lastReceivedBidUtility = 0;
|
---|
| 57 | currentVote = Vote.REJECT;
|
---|
| 58 | currentRound = 0;
|
---|
| 59 | }
|
---|
| 60 |
|
---|
| 61 | /**
|
---|
| 62 | * When this class is called, it is expected that the Party chooses one of
|
---|
| 63 | * the actions from the possible action list and returns an instance of the
|
---|
| 64 | * chosen action. This class is only called if this {@link NegotiationParty}
|
---|
| 65 | * is in the
|
---|
| 66 | * {@link negotiator.protocol .DefaultProtocol#getRoundStructure(java.util.List, negotiator.session.Session)}
|
---|
| 67 | * .
|
---|
| 68 | *
|
---|
| 69 | * @param possibleActions
|
---|
| 70 | * List of all actions possible.
|
---|
| 71 | * @return The chosen action
|
---|
| 72 | */
|
---|
| 73 | @Override
|
---|
| 74 | public Action chooseAction(List<Class<? extends Action>> possibleActions) {
|
---|
| 75 | return new VoteForOfferAcceptance(getPartyId(), currentVote);
|
---|
| 76 | }
|
---|
| 77 |
|
---|
| 78 | /**
|
---|
| 79 | * This method is called when an observable action is performed. Observable
|
---|
| 80 | * actions are defined in
|
---|
| 81 | * {@link MultilateralProtocol#getActionListeners(java.util.List)}
|
---|
| 82 | *
|
---|
| 83 | * @param sender
|
---|
| 84 | * The initiator of the action
|
---|
| 85 | * @param arguments
|
---|
| 86 | * The action performed
|
---|
| 87 | */
|
---|
| 88 | @Override
|
---|
| 89 | public void receiveMessage(AgentID sender, Action arguments) {
|
---|
| 90 | if (arguments instanceof OfferForVoting) {
|
---|
| 91 | lastReceivedBidUtility = getUtility(((OfferForVoting) arguments).getBid());
|
---|
| 92 | if (lastReceivedBidUtility >= lastAcceptedBidUtility) {
|
---|
| 93 | currentVote = Vote.ACCEPT;
|
---|
| 94 | } else {
|
---|
| 95 | double temperature = getNormalizedRoundOrTimeValue();
|
---|
| 96 |
|
---|
| 97 | double probability = pow(E, ((lastReceivedBidUtility - lastAcceptedBidUtility) / temperature));
|
---|
| 98 | currentVote = probability > rand.nextDouble() ? Vote.ACCEPT : Vote.REJECT;
|
---|
| 99 | }
|
---|
| 100 | } else if (arguments instanceof InformVotingResult) {
|
---|
| 101 | if (((InformVotingResult) arguments).getVotingResult() == Vote.ACCEPT) {
|
---|
| 102 | lastAcceptedBidUtility = lastReceivedBidUtility;
|
---|
| 103 | }
|
---|
| 104 | currentRound++;
|
---|
| 105 | }
|
---|
| 106 | }
|
---|
| 107 |
|
---|
| 108 | /**
|
---|
| 109 | * returns the highest of normalized time and round values
|
---|
| 110 | *
|
---|
| 111 | * @return a value between 0.0 and 1.0
|
---|
| 112 | */
|
---|
| 113 | private double getNormalizedRoundOrTimeValue() {
|
---|
| 114 | // return the most urgent value
|
---|
| 115 | return max(getNormalizedRoundValue(), getNormalizedTimeValue());
|
---|
| 116 | }
|
---|
| 117 |
|
---|
| 118 | /**
|
---|
| 119 | * Gets the normalized current round between 0.0 and 1.0
|
---|
| 120 | *
|
---|
| 121 | * @return A value between 0.0 and 1.0 or 0.0 if no round deadline given
|
---|
| 122 | */
|
---|
| 123 | private double getNormalizedRoundValue() {
|
---|
| 124 | if (getDeadlines().getType() == DeadlineType.ROUND) {
|
---|
| 125 | return ((double) currentRound / (double) getDeadlines().getValue());
|
---|
| 126 | } else {
|
---|
| 127 | return 0d;
|
---|
| 128 | }
|
---|
| 129 | }
|
---|
| 130 |
|
---|
| 131 | /**
|
---|
| 132 | * Gets the normalized current time, which runs from 0.0 to 1.0
|
---|
| 133 | *
|
---|
| 134 | * @return A value between 0.0 and 1.0 which defaults to 0.0 if no time
|
---|
| 135 | * deadline set
|
---|
| 136 | */
|
---|
| 137 | private double getNormalizedTimeValue() {
|
---|
| 138 | return timeline == null ? 0d : timeline.getTime();
|
---|
| 139 | }
|
---|
| 140 |
|
---|
| 141 | @Override
|
---|
| 142 | public String getDescription() {
|
---|
| 143 | return "Annealer Party";
|
---|
| 144 | }
|
---|
| 145 |
|
---|
| 146 | }
|
---|