DEV Community

Philippe Vaillancourt
Philippe Vaillancourt

Posted on • Originally published at blog.snowfrog.dev on

The Spy of Testadel: A Medieval Testing Tale

The Spy of Testadel: A Medieval Testing Tale

The year is 1453. After 116 years of war, your kingdom, Testadel, is finally at peace. Now is the time to rebuild and strengthen your relationships with your neighbours. To the North lies the fiefdom of Lord PureFunction, who fought valiantly at your side during the war. In an effort to establish stronger commercial ties, you both sign a treaty whereby you promise to supply his keep with as many sheep as you can every month, in exchange for 15 pieces of gold per sheep.

class LordPureFunctionsKeep {
  priceOfOneSheep = 15;

  calculatePaymentForSheep(sheep: number): number {    
    return sheep.length * this.priceOfOneSheep;
  }
}
Enter fullscreen mode Exit fullscreen mode

After a few months, as was agreed by both parties, an independent auditor travels to Lord PureFunction's keep to make sure that the commercial agreement is being fulfilled properly.

describe('LordPureFunctionsKeep'() => {
  test('should pay 300 gold pieces when supplied with 20 sheep', () => {
    const keep = new LordPureFunctionsKeep();

    const payment = keep.calculatePaymentForSheep(20);

    expect(payment).toBe(300);
  });
});
Enter fullscreen mode Exit fullscreen mode

Upon completion, the auditor reports that Lord PureFunction was very accommodating. A true paragon of virtue, PureFunction opened up its books willingly and complied happily with every request from the auditor. "Usually", the auditor says, "people try to hide things from me, but not so with Lord PureFunction. You are lucky to have such a transparent and trustworthy ally".

Happy with the auditor's findings and secure in the knowledge that you have a strong relationship with your ally to the North, you turn your attention to the South. There lie the ravaged lands of Sir SideEffects.

The Spy of Testadel: A Medieval Testing Tale
Photo by udit saptarshi / Unsplash

Sir SideEffects had the misfortune of taking the other side, the losing side, during the recently ended war. The peace treaty that ended the war, and that he signed reluctantly, imposes serious limitations on the amount of weapons he is allowed to stockpile and forbids him from having any contacts with his previous allies.

It is agreed that Sir SideEffects will provide every month to your envoy a full account of his weapons stock.

class SirSideEffectsCamp {
  private armoury: Weapons[] = [] ;
  private scribe = new Scribe();

  reportWeaponsStock(): number {
    const report = this.scribe.writeWeaponsReport(this.armoury);
    return report;
  }
}
Enter fullscreen mode Exit fullscreen mode

<!--kg-card-end: markdown--><!--kg-card-begin: markdown-->

describe('SirSideEffectsCamp'() => {
  test('should report having less than 50 weapons', () => {
    const camp = new SirSideEffectsCamp();

    const weaponsStock = camp.reportWeaponsStock();

    expect(weaponsStock).toBeLessThan(50);
  });
});
Enter fullscreen mode Exit fullscreen mode

For the first few months, Sir SideEffects seems to abide by the terms of his surrender. Every month your envoy shows up at the gate, asks for a report, and every month Sir SideEffects has a piece of paper attached to a rock thrown over the rampart, detailing the level of his weapons stock. The amount varies a little every month but is always below the agreed ceiling.

The Spy of Testadel: A Medieval Testing Tale

Everything seems under control until one day your envoy asks to speak with you about a matter that seems to have him worried. "My lord, I fear that Sir SideEffects might be up to no good. Indeed, this is the fourth month in a row that he has reported exactly 6 weapons. Something's not right; the amount always used to vary before".

class SirSideEffectsCamp {
  private armoury = Weapons[] = [];
  private scribe = new Scribe();

  reportWeaponsStock(): number { 
    const report = this.scribe.writeWeaponsReport(this.armoury);
    return 6; // <- Now reports 6 weapons
  }
}
Enter fullscreen mode Exit fullscreen mode

Taking this matter to heart, you convene a meeting of your councillors to address the situation with Sir SideEffects. After debating for the better part of the day, everyone agrees that although it is highly suspicious that the reports for the past fours months have all been the same, it is not in itself a proof of foul play. More information needs to be gathered. But how? Sir SideEffects keeps the gate to his camp closed to outsiders and it is impossible to see what is going on inside.

One of your councillors speaks up. "Sir, we need to send a spy. Someone crafty enough to find a way in without being recognized and raising the alarm. The spy could check the weapons store and report back. Then we'll know for sure."

The Spy of Testadel: A Medieval Testing Tale

And so it is decided, a spy will be sent to Sir SideEffects' camp, with the difficult mission of infiltrating the camp and gathering vital information. Thankfully, you know just the man for the job. Your court jester, aptly nicknamed Jest, is a master of disguise and despite what some might think, an intelligent and resourceful man.

Later, as you go over the mission's plan with Jest, he points out a problem with the plan. If he is to somehow infiltrate the camp under the guise of one of its inhabitants, an opportunity for a swap must be created. At the moment, such an exchange is impossible as all of the camp's residents remain inside day and night.

"Leave that up to me," you say, "I know just what to do. I'll send forged orders to Sir SideEffects' scribe requiring him to leave the camp for the day. Upon his return, you can intercept him a short distance from the gate and take his place. You should be able to walk right through the front gate if your disguise is convincing enough."

As Jest makes preparation for his departure, you arrange for the forged orders to be sent to Sir SideEffects' scribe.

class SirSideEffectsCamp {
  private armoury = Weapons[];
  constructor(private scribe: Scribe) { } // <- Scribe now passes through gate

  reportWeaponsStock(): number { 
    const report = this.scribe.writeWeaponsReport(this.armoury);
    return 6;
  }
}
Enter fullscreen mode Exit fullscreen mode

The next day, exactly as planned, Jest lies in wait by the wayside, about half a league from SideEffects' camp. As the sun begins to set, he finally sights his target. The scribe, back from surveying Sir SideEffects' fields, can already taste the turnip and cabbage soup waiting for him at home. Lost in his thoughts, he pays no attention to his surroundings. As he passes by the bushes where Jest had been lying in wait, he receives a massive blow to the head and immediately falls to the ground, unconscious.

Jest drags the scribe's limp body into the bushes, removes his clothing and adorns himself with the lavish clothes and accoutrement.

describe('SirSideEffectsCamp'() => {
  test('should report having less than 50 weapons', () => {
    // Our spy mimicks the scribe's ability to 'writeWeaponsReport'
    // but makes sure to keep a copy of the report in his pocket
    const spyScribe = {
      writeWeaponsReport(armoury: Weapons[]) {
        const report = armoury.length;
        spyScribe.pocket = report; // <- Keep copy of report in pocket
        return report;
      }
    }
    const camp = new SirSideEffectsCamp();

    const weaponsStock = camp.reportWeaponsStock();

    expect(weaponsStock).toBeLessThan(50);
  });
});
Enter fullscreen mode Exit fullscreen mode

All that is left now is to walk through the front gate and take the scribe's place.

describe('SirSideEffectsCamp'() => {
  test('should report having less than 50 weapons', () => {
    const spyScribe = {
      writeWeaponsReport(armoury: Weapons[]) {
        const report = this.armoury.length;
        spyScribe.pocket = report;
        return report;
      }
    }
    const camp = new SirSideEffectsCamp(spyScribe); // <- Spy infiltrates the camp

    const weaponsStock = camp.reportWeaponsStock();

    expect(weaponsStock).toBeLessThan(50);
  });
});
Enter fullscreen mode Exit fullscreen mode

The next morning, your envoy shows up at the gate of Sir SideEffects' camp and asks for a report like he did every month since the end of the war. This time though, your spy is in place inside the camp and is able to observe the weapons stock from the inside.

describe('SirSideEffectsCamp'() => {
  test('should report having less than 50 weapons', () => {
    const spyScribe = {
      writeWeaponsReport(armoury: Weapons[]) {
        const report = this.armoury.length;
        spyScribe.pocket = report;
        return report;
      }
    }
    const camp = new SirSideEffectsCamp(spyScribe);

    camp.reportWeaponsStock();

    expect(spyScribe.pocket).toBeLessThan(50); // <- Get spy's report
  });
});
Enter fullscreen mode Exit fullscreen mode

Later that day, your spy manages to sneak out and report his findings. Although the report given by Sir SideEffects once again said that he only had 6 weapons in his armory, Jest reports that he in fact has close to 100 weapons in stock and that he has been willfully giving false reports.

What's more, while in the camp, Jest has heard a troubling rumour. Word around the drinking well is that Sir SideEffects has taken to dark magic. Apparently, the rumour goes, Sir SideEffects is in possession of a magical artifact that allows him to communicate directly with anyone, however far removed from the camp they may be. The rumour even goes as far as to say that the artifact - some say it is called The ConsoleTM - would allow him to communicate with mystical beings from another realm.

The Spy of Testadel: A Medieval Testing Tale

Though this seems a little far-fetched, you do suspect Sir SideEffects is using the artifact to communicate with his old ally, Sir EvilScreen.

class SirSideEffectsCamp {
  private armoury = Weapons[];
  constructor(private scribe: Scribe) { }

  reportWeaponsStock(): number { 
    const report = this.scribe.writeWeaponsReport(this.armoury);
    console.log(`We now have ${report} weapons, death to Testadel!`); // <-
    return 6;
  }
}
Enter fullscreen mode Exit fullscreen mode

If only there was a way to intercept the messages being sent through this magical object. "My Lord," a croaked voice is heard from the back of the hall as a wizened old man steps forward, "my name is Sir Mockalot and through my travels and adventures, I have acquired an object that I believe may be of assistance".

Sir Mockalot explains that he possesses an artifact called the The MocksoleTM. "It is said to have been forged together with The ConsoleTM and to possess very similar properties. If we could substitute The ConsoleTM with our own artifact, we would be the ones receiving Sir SideEffects' messages and he would be none the wiser".

As Sir Mockalot, accompanied by Jest, your trusty jester / spy, goes to his chamber to prepare The MocksoleTM for delivery to Sir SideEffects camp,

describe('SirSideEffectsCamp'() => {
  test('should not be sending treasonous messages', () => {
    const theMocksole = { // <- Preparing the Mocksole
      log: jest.fn()
    }
  });
});
Enter fullscreen mode Exit fullscreen mode

you set to writing an accompanying note.

"Dear Sir SideEffects, I was not able to hear your last message through The ConsoleTM as the sound was all garbled. Please find attached to this note, a replacement artifact to be used in its stead. Sincerely, Lord EvilScreen."

The next day, The MocksoleTM is found at the camp's gate and brought in. Sir SideEffects reads the note and complies with its message.

class SirSideEffectsCamp {
  private armoury = Weapons[];
  constructor(private scribe: Scribe, private artifact: Console) { } // <- A console-like artifact at the gate

  reportWeaponsStock(): void { 
    const report = this.scribe.writeWeaponsReport(this.armoury);
    artifact.log(`We now have ${report} weapons, death to Testadel!`); // <- Let's use it instead of the old console.
    return 6;
  }
}
Enter fullscreen mode Exit fullscreen mode

Everything is now in place to intercept the messages, all you need to do is stand by and listen.

describe('SirSideEffectsCamp'() => {
  test('should not be sending treasonous messages', () => {
    const theMocksole = {
      log: jest.fn()
    }
    const treasonousTalk = 'death to Testadel!'
    const camp = new SirSideEffectsCamp(spyScribe, theMocksole);

    camp.reportWeaponsStock();

    expect(theMocksole.mock.calls[0][0]).not.stringMatching(treasonousTalk);
  });
});
Enter fullscreen mode Exit fullscreen mode

Horror! It is as you feared, Sir SideEffects is planning a rebellion. Thanks to the valiant efforts of your subjects you are made aware of this fact before it is too late and can make preparations. Peace will have unfortunately been short-lived.

Top comments (0)