<!DOCTYPE html>
<html>
<head>
<style type='text/css'>
body { background: red; }
#zoom { margin-left: 680px; }
</style>
<script>

// This function has problems with paths that include multiple subpaths.
// First, if the boundaries of two subpaths are close, and
// the stroke outside of one would be inside of the other, then we can
// get artifacts.  
// Second, when one subpath is entirely enclosed inside the other and
// the windings are the same, then both the inside and outside of the
// inner subpath are "inside".  So should we stroke both sides?
// That is what this method does.  It seems kind of logical, but not really
// what we'd want.
//
// The upshot is that strokeInside() is not a fully-general replacement for
// stroke().  When multiple subpaths are required they may need to be 
// stroked individually rather than as a group.
function strokeInside(c) {
    c.save();
    c.clip();
    c.lineWidth *= 2;
    c.stroke();
    c.restore();
}

function strokeOutside(c) {
    // Back up the entire canvas to an offscreen canvas
    var offscreen = document.createElement("canvas");
    offscreen.width = c.canvas.width;
    offscreen.height = c.canvas.height;
    var offC = offscreen.getContext('2d');
    offC.drawImage(c.canvas, 0, 0);

    // Save current state
    c.save();

    // Stroke the shape with double linewidth
    c.lineWidth *= 2;
    c.stroke();

    // Now set the clipping region and restore the interior of the path
    // to its original state before the stroke
    c.clip();
    c.globalCompositeOperation = "copy";  // So we don't blend colors
    c.setTransform(1,0,0,1,0,0);          // Back to default coordinate system
    c.drawImage(offscreen, 0, 0);         // Copy the image back

    // Restore the graphics state
    c.restore();

    // Deallocate offscreen pixels?
    offscreen.width = offscreen.height = 0;
}

function shape(c) {
    c.beginPath();

    c.moveTo(100,10);
    c.lineTo(140,170);
    c.lineTo(10,40);
    c.lineTo(150,40);
    c.lineTo(60,175);
    c.closePath();

    c.rect(0,200,200,150);

    c.moveTo(65,265);
    c.arc(65,265,50,0,Math.PI/3,true);
    c.closePath();

    c.moveTo(130,280);
    c.arc(130,280,50,0,Math.PI/3,false);
    c.closePath();

    c.fill();
    c.save();
    c.lineWidth = arguments[1] || 0;
    c.strokeStyle = c.fillStyle;
    c.stroke();
    c.restore();
}

function demo() {
    var canvas  = document.getElementById("canvas");
    var c = canvas.getContext('2d');

    c.fillStyle = "rgba(246,246,246,1)";
    c.strokeStyle = "rgba(150,150,150,1)";
    c.lineWidth = 8;

  
    c.translate(20,20);
    shape(c);
    c.stroke();       // Regular stroke


    // What does it mean to stroke inside a path when subpaths overlap?
    // if both sides of a subpath are inside a containing path, should
    // both sides be stroked?
    c.translate(250, 0);
    shape(c);
    strokeInside(c);  // Stroke inside the path

    c.translate(250, 0);
    shape(c);
    strokeOutside(c); // Stroke outside the path

                c.translate(250, 0);
    shape(c, 1);
    strokeOutside(c); // Stroke outside the path with a 1px stroke on the shape, only fixes firefox
}

window.onload = demo;
</script>
</head>
<body>
<canvas id="canvas" width=980 height=450></canvas>
<p>
<img id='zoom' src='%3D' alt=''/>
</p>
</body>
</html>