#include <bits/stdc++.h>
using namespace std;
#define ll long long
#define double long double
void fileIO()
{
ios_base::sync_with_stdio(0);
cin.tie(0);
cout.tie(0);
freopen("raydist.in", "r", stdin);
freopen("raydist.out", "w", stdout);
#ifndef ONLINE_JUDGE
freopen("input.txt", "r", stdin);
freopen("output.txt", "w", stdout);
#endif
}
typedef double T;
typedef complex<T> pt;
#define x real()
#define y imag()
#define vec(a, b) (pt)(b - a)
#define M_PI 3.14159265358979323846
#define eps 1e-9
#define dot(a, b) ((conj(a) * (b)).real())
#define cross(a, b) ((conj(a) * (b)).imag())
#define perp(p) \
(pt) { -p.y, p.x } // the rotation by 90◦
#define sq(a) a.x *a.x + a.y *a.y
#define orient(a, b, c) cross(b - a, c - a)
struct Line
{
pt v;
T c;
// from vector and offset
Line(pt v, T c) : v(v), c(c) {}
// ax + by = c
Line(T a, T b, T c) : v({-b, a}), c(c) {}
// from two points
Line(pt a, pt b) : v(b - a), c(cross(v, a)) {}
// left if positive, right if negative, on the line otherwise
T side(pt p) { return cross(v, p) - c; }
// distance from p to the line
double dist(pt p) { return abs(side(p)) / abs(v); }
double sqDist(pt p) { return side(p) * side(p) / (double)sq(v); }
Line perpThrough(pt p) { return {p, p + perp(v)}; }
// sort points on line
bool cmpProj(pt p, pt q) { return dot(v, p) < dot(v, q); }
// translate a line by direction vector t, only c changes
Line translate(pt t) { return {v, c + cross(v, t)}; }
// shift the line left by distance dist, direction vector t is (dist / abs(v)) * perp(v)
Line shiftLeft(double dist) { return {v, c + dist * abs(v)}; }
// intersection point between two lines
bool inter(Line l, pt &out)
{
T d = cross(v, l.v);
if (d == 0)
return false;
out = (l.v * c - v * l.c) / d; // requires floating-point coordinates
return true;
}
// projection of a point P on a line l is the point on l that is closest to P.
pt proj(pt p) { return p - perp(v) * side(p) / sq(v); }
// The reflection of point P by line l is the point on the other side of l that is at the same distance and has the same orthogonal projection.
pt refl(pt p) { return p - perp(v) * (2.0 * side(p) / sq(v)); }
/*
l_ext(l1,l2)▲
│
│
│ l2
xx │ xxxxxxxx
xxxx │ xxxxxxx
xxx │ xxxxxx
xxx │ xxx x
xx │ xxx
xxx │ xxx l_int(l1,l2)
─────────────────xxxxx─────────────────►
xxxxxxxxx
xxx │ xx
xxx │ xxx
xxx │ xxxx xx
│ xxxxxxx
│ xxxxxxx
│ xxxxxxxxxx
│ xxxxxxxxxxx
l1
*/
Line bisector(Line l, bool interior)
{
assert(cross(v, l.v) != 0); // l1 and l2 cannot be parallel!
double sign = interior ? 1 : -1;
return {l.v / abs(l.v) + v / abs(v) * sign,
l.c / abs(l.v) + c / abs(v) * sign};
}
};
// checks if a point lies on disk formed by diameter [a,b]
static bool inDisk(pt a, pt b, pt p) { return dot(a - p, b - p) <= 0; }
// check if point is on the segment [a,b]
static bool onSegment(pt a, pt b, pt p) { return orient(a, b, p) == 0 && inDisk(a, b, p); }
// one point intersection of two segments
bool properInter(pt a, pt b, pt c, pt d, pt &out)
{
double oa = orient(c, d, a),
ob = orient(c, d, b),
oc = orient(a, b, c),
od = orient(a, b, d);
// Proper intersection exists if opposite signs
if (oa * ob < 0 && oc * od < 0)
{
out = (a * ob - b * oa) / (ob - oa);
return true;
}
return false;
}
// comparable to create set of points
struct cmpX
{
bool operator()(pt a, pt b) const
{
return make_pair(a.x, a.y) < make_pair(b.x, b.y);
}
};
// the intersection between two points
set<pt, cmpX> intersections(pt a, pt b, pt c, pt d)
{
pt out;
if (properInter(a, b, c, d, out))
return {out}; // only one intersection
set<pt, cmpX> s;
if (onSegment(c, d, a))
s.insert(a);
if (onSegment(c, d, b))
s.insert(b);
if (onSegment(a, b, c))
s.insert(c);
if (onSegment(a, b, d))
s.insert(d);
return s;
}
double segmentPointDistance(pt a, pt b, pt p)
{
if (a != b)
{
Line l(a, b);
if (l.cmpProj(a, p) && l.cmpProj(p, b)) // if closest to projection
return l.dist(p); // output distance to Line
}
return min(abs(p - a), abs(p - b)); // otherwise distance to A or B
}
double segmentSegmentDistance(pt a, pt b, pt c, pt d)
{
pt dummy;
if (properInter(a, b, c, d, dummy))
return 0;
return min({segmentPointDistance(a, b, c), segmentPointDistance(a, b, d),
segmentPointDistance(c, d, a), segmentPointDistance(c, d, b)});
}
double rayPointDistance(pt a, pt b, pt p)
{
if (a != b)
{
Line l(a, b);
if (l.cmpProj(a, p)) // if closest to projection
return l.dist(p); // output distance to Line
}
return abs(p - a); // otherwise distance to A
}
bool intersectSegments(pt a, pt b, pt c, pt d, pt &intersect)
{
double d1 = cross(vec(b, a), vec(c, d)), d2 = cross(vec(c, a), vec(c, d)), d3 = cross(vec(b, a), vec(c, a));
if (fabs(d1) < eps)
return false; /// parllel or identical
double t1 = d2 / d1, t2 = d3 / d1;
intersect = a + (b - a) * t1;
/// segment: [0, 1]
/// ray: [0, INF]
/// line: [-INF, INF]
/// make sure that eps is not too accurate
if (t1 < -eps or t2 < -eps)
return false;
return true;
}
double rayRayDistance(pt a, pt b, pt c, pt d)
{
pt dummy;
if (intersectSegments(a, b, c, d, dummy))
return 0;
return min(rayPointDistance(a, b, c), rayPointDistance(c, d, a));
}
void Striker()
{
int a, b;
cin >> a >> b;
pt p1(a, b);
cin >> a >> b;
pt p2(a, b);
cin >> a >> b;
pt p3(a, b);
cin >> a >> b;
pt p4(a, b);
cout << fixed << setprecision(12) << rayRayDistance(p1, p2, p3, p4);
}
signed main()
{
fileIO();
int _t = 1;
// cin >> _t;
for (int i = 0; i < _t; ++i)
{
Striker();
}
}
#include <bits/stdc++.h>
using namespace std;

#define ll long long
#define double long double
void fileIO()
{
    ios_base::sync_with_stdio(0);
    cin.tie(0);
    cout.tie(0);
    freopen("raydist.in", "r", stdin);
    freopen("raydist.out", "w", stdout);
#ifndef ONLINE_JUDGE
    freopen("input.txt", "r", stdin);
    freopen("output.txt", "w", stdout);
#endif
}

typedef double T;
typedef complex<T> pt;
#define x real()
#define y imag()
#define vec(a, b) (pt)(b - a)
#define M_PI 3.14159265358979323846
#define eps 1e-9
#define dot(a, b) ((conj(a) * (b)).real())
#define cross(a, b) ((conj(a) * (b)).imag())
#define perp(p) \
    (pt) { -p.y, p.x } // the rotation by 90◦
#define sq(a) a.x *a.x + a.y *a.y
#define orient(a, b, c) cross(b - a, c - a)
struct Line
{
    pt v;
    T c;

    // from vector and offset
    Line(pt v, T c) : v(v), c(c) {}
    // ax + by = c
    Line(T a, T b, T c) : v({-b, a}), c(c) {}
    // from two points
    Line(pt a, pt b) : v(b - a), c(cross(v, a)) {}
    // left if positive, right if negative, on the line otherwise
    T side(pt p) { return cross(v, p) - c; }
    // distance from p to the line
    double dist(pt p) { return abs(side(p)) / abs(v); }
    double sqDist(pt p) { return side(p) * side(p) / (double)sq(v); }
    Line perpThrough(pt p) { return {p, p + perp(v)}; }
    // sort points on line
    bool cmpProj(pt p, pt q) { return dot(v, p) < dot(v, q); }
    // translate a line by direction vector t, only c changes
    Line translate(pt t) { return {v, c + cross(v, t)}; }
    // shift the line left by distance dist, direction vector t is (dist / abs(v)) * perp(v)
    Line shiftLeft(double dist) { return {v, c + dist * abs(v)}; }
    // intersection point between two lines
    bool inter(Line l, pt &out)
    {
        T d = cross(v, l.v);
        if (d == 0)
            return false;
        out = (l.v * c - v * l.c) / d; // requires floating-point coordinates
        return true;
    }
    // projection of a point P on a line l is the point on l that is closest to P.
    pt proj(pt p) { return p - perp(v) * side(p) / sq(v); }
    // The reflection of point P by line l is the point on the other side of l that is at the same distance and has the same orthogonal projection.
    pt refl(pt p) { return p - perp(v) * (2.0 * side(p) / sq(v)); }

    /*
        l_ext(l1,l2)▲
                   │
                   │
                   │               l2
     xx            │        xxxxxxxx
      xxxx         │        xxxxxxx
          xxx      │        xxxxxx
            xxx    │     xxx   x
              xx   │   xxx
               xxx │ xxx                 l_int(l1,l2)
─────────────────xxxxx─────────────────►
               xxxxxxxxx
             xxx   │    xx
           xxx     │      xxx
         xxx       │        xxxx   xx
                   │           xxxxxxx
                   │            xxxxxxx
                   │          xxxxxxxxxx
                   │          xxxxxxxxxxx

                                 l1
    */

    Line bisector(Line l, bool interior)
    {
        assert(cross(v, l.v) != 0); // l1 and l2 cannot be parallel!
        double sign = interior ? 1 : -1;
        return {l.v / abs(l.v) + v / abs(v) * sign,
                l.c / abs(l.v) + c / abs(v) * sign};
    }
};

// checks if a point lies on disk formed by diameter [a,b]
static bool inDisk(pt a, pt b, pt p) { return dot(a - p, b - p) <= 0; }
// check if point is on the segment [a,b]
static bool onSegment(pt a, pt b, pt p) { return orient(a, b, p) == 0 && inDisk(a, b, p); }

// one point intersection of two segments
bool properInter(pt a, pt b, pt c, pt d, pt &out)
{
    double oa = orient(c, d, a),
           ob = orient(c, d, b),
           oc = orient(a, b, c),
           od = orient(a, b, d);
    // Proper intersection exists if opposite signs
    if (oa * ob < 0 && oc * od < 0)
    {
        out = (a * ob - b * oa) / (ob - oa);
        return true;
    }
    return false;
}

// comparable to create set of points
struct cmpX
{
    bool operator()(pt a, pt b) const
    {
        return make_pair(a.x, a.y) < make_pair(b.x, b.y);
    }
};

// the intersection between two points
set<pt, cmpX> intersections(pt a, pt b, pt c, pt d)
{
    pt out;
    if (properInter(a, b, c, d, out))
        return {out}; // only one intersection
    set<pt, cmpX> s;
    if (onSegment(c, d, a))
        s.insert(a);
    if (onSegment(c, d, b))
        s.insert(b);
    if (onSegment(a, b, c))
        s.insert(c);
    if (onSegment(a, b, d))
        s.insert(d);
    return s;
}

double segmentPointDistance(pt a, pt b, pt p)
{
    if (a != b)
    {
        Line l(a, b);
        if (l.cmpProj(a, p) && l.cmpProj(p, b)) // if closest to projection
            return l.dist(p);                   // output distance to Line
    }
    return min(abs(p - a), abs(p - b)); // otherwise distance to A or B
}

double segmentSegmentDistance(pt a, pt b, pt c, pt d)
{
    pt dummy;
    if (properInter(a, b, c, d, dummy))
        return 0;
    return min({segmentPointDistance(a, b, c), segmentPointDistance(a, b, d),
                segmentPointDistance(c, d, a), segmentPointDistance(c, d, b)});
}
double rayPointDistance(pt a, pt b, pt p)
{
    if (a != b)
    {
        Line l(a, b);
        if (l.cmpProj(a, p))  // if closest to projection
            return l.dist(p); // output distance to Line
    }
    return abs(p - a); // otherwise distance to A
}

bool intersectSegments(pt a, pt b, pt c, pt d, pt &intersect)
{
    double d1 = cross(vec(b, a), vec(c, d)), d2 = cross(vec(c, a), vec(c, d)), d3 = cross(vec(b, a), vec(c, a));

    if (fabs(d1) < eps)
        return false; /// parllel or identical

    double t1 = d2 / d1, t2 = d3 / d1;
    intersect = a + (b - a) * t1;

    /// segment: [0, 1]
    /// ray: [0, INF]
    /// line: [-INF, INF]

    /// make sure that eps is not too accurate
    if (t1 < -eps or t2 < -eps)
        return false;

    return true;
}

double rayRayDistance(pt a, pt b, pt c, pt d)
{
    pt dummy;
    if (intersectSegments(a, b, c, d, dummy))
        return 0;
    return min(rayPointDistance(a, b, c), rayPointDistance(c, d, a));
}

void Striker()
{
    int a, b;
    cin >> a >> b;
    pt p1(a, b);
    cin >> a >> b;
    pt p2(a, b);
    cin >> a >> b;
    pt p3(a, b);
    cin >> a >> b;
    pt p4(a, b);
    cout << fixed << setprecision(12) << rayRayDistance(p1, p2, p3, p4);
}

signed main()
{
    fileIO();
    int _t = 1;
    // cin >> _t;
    for (int i = 0; i < _t; ++i)
    {
        Striker();
    }
}
