#include <vector>
#include <cassert>
#include <utility>
#include <algorithm>
#include <cstdio>
#include <string>
#include <tuple>
struct Interval {
int l, r;
int length() const { return r-l+1; }
};
long long pow10[17], ones[17];
int getBestCount(int L, int R, int power, int logPower, int D) {
if(power == 1) {
if(L <= D && D <= R) {
return 1;
}else {
return 0;
}
}
if(L == 0) {
int cnt = 0;
while(ones[cnt + 1] * D <= R) cnt += 1;
return cnt;
}
if(R == power - 1) {
int cnt = logPower, dep = 0;
while(L <= ones[cnt] * D + ones[dep] * 9 * pow10[cnt]) {
cnt -= 1;
dep += 1;
}
return cnt;
}
const int L0 = L / power, R0 = R / power;
if(L0 == R0) {
int childCnt = getBestCount(L % power, R % power, power / 10, logPower - 1, D);
if(D == L0) childCnt ++;
return childCnt;
}
int count = 0;
for(int digit = L0; digit <= R0; digit++) {
int childCount;
if(digit == L0) {
childCount = getBestCount(L % power, power - 1, power / 10, logPower - 1, D);
}else if(L0 < digit && digit < R0) {
childCount = logPower;
}else {
childCount = getBestCount(0, R % power, power / 10, logPower - 1, D);
}
int nextCount = childCount + (digit == D ? 1 : 0);
if(nextCount < count) continue;
if(nextCount > count) {
count = nextCount;
}
}
return count;
}
std::vector<Interval> genBestIntervals(int L, int R, int power, int logPower, int D) {
if(power == 1) {
if(L <= D && D <= R) {
return std::vector<Interval>(1, Interval{D, D});
}else {
return std::vector<Interval>(1, Interval{L, R});
}
}
const int L0 = L / power, R0 = R / power;
if(L0 == R0) {
auto child = genBestIntervals(L % power, R % power, power / 10, logPower - 1, D);
if(L0 > 0) for(auto &intv : child) intv.l += L0 * power, intv.r += L0 * power;
return child;
}
int count = getBestCount(L, R, power, logPower, D);
std::vector<Interval> intervals;
for(int digit = L0; digit <= R0; digit++) {
int childCount;
if(digit == L0) {
childCount = getBestCount(L % power, power - 1, power / 10, logPower - 1, D);
}else if(L0 < digit && digit < R0) {
childCount = logPower;
}else {
childCount = getBestCount(0, R % power, power / 10, logPower - 1, D);
}
int nextCount = childCount + (digit == D ? 1 : 0);
if(nextCount < count) continue;
assert(nextCount == count);
std::vector<Interval> childIntervals;
if(digit == L0) {
childIntervals = genBestIntervals(L % power, power - 1, power / 10, logPower - 1, D);
}else if(L0 < digit && digit < R0) {
childIntervals.push_back({(power - 1) / 9 * D, (power - 1) / 9 * D});
}else {
childIntervals = genBestIntervals(0, R % power, power / 10, logPower - 1, D);
}
for(Interval intv : childIntervals) {
intv.l += digit * power;
intv.r += digit * power;
intervals.push_back(intv);
}
}
return intervals;
}
struct Event {
int number, who, inserted;
};
int main() {
#ifdef IN_MY_COMPUTER
freopen("example.in.txt", "r", stdin);
#endif
pow10[0] = 1;
for(int i = 1; i < 17; i++) pow10[i] = pow10[i-1] * 10, ones[i] = ones[i-1] * 10 + 1;
int T; scanf("%d", &T);
for(int tc = 1; tc <= T; tc++) {
int A, B, C, D; scanf("%d%d%d%d", &A,&B,&C,&D);
int logPower = std::to_string(B).length() - 1;
int power = 1; for(int rep = 0; rep < logPower; rep++) power *= 10;
auto CIntervals = genBestIntervals(A, B, power, logPower, C);
auto DIntervals = genBestIntervals(A, B, power, logPower, D);
std::vector<Event> events;
for(auto it : CIntervals) events.push_back({it.l, 0, +1}), events.push_back({it.r+1, 0, -1});
for(auto it : DIntervals) events.push_back({it.l, 1, +1}), events.push_back({it.r+1, 1, -1});
std::sort(events.begin(), events.end(), [&](const Event &e1, const Event &e2) { return e1.number < e2.number; });
events.push_back({B+1, -1, -1});
int bestAnswer = -1;
long long bestProbability = -1;
auto updateAnswer = [&bestAnswer, &bestProbability](int pos, long long prob) {
if(prob > bestProbability) {
bestProbability = prob;
bestAnswer = pos;
}else if(prob == bestProbability && bestAnswer > pos) {
bestAnswer = pos;
}
};
long long totalLength[] = {0, 0};
for(auto &it : CIntervals) totalLength[0] += it.length();
for(auto &it : DIntervals) totalLength[1] += it.length();
int alive[] = {0, 0};
long long totalCommonCases = 0;
for(size_t _idx = 0; _idx+1 < events.size(); _idx++) {
Event ¤t = events[_idx];
Event &next = events[_idx+1];
alive[current.who] = alive[current.who] + current.inserted;
if(current.number < next.number || next.who < 0) {
if(alive[0] > 0 && alive[1] > 0) {
totalCommonCases += next.number - current.number;
}
}
}
alive[0] = 0;
alive[1] = 0;
long long leftLength[] = {0, 0};
long long rightLength[] = {totalLength[0], totalLength[1]};
for(size_t _idx = 0; _idx+1 < events.size(); _idx++) {
Event ¤t = events[_idx];
Event &next = events[_idx+1];
alive[current.who] = alive[current.who] + current.inserted;
if(current.number < next.number || next.who < 0) {
const long long length = (next.number - current.number);
for(int i= 0; i < 2; i++) {
if(alive[i] > 0) rightLength[i] = rightLength[i] - length;
}
// [5, 8) length=3
// 5, 6 (7)
// (5) 6 7
auto update = [&](int x) {
long long prob =
3 * (leftLength[0] + x * alive[0]) * (rightLength[1] + (length - 1 - x) * alive[1]) +
3 * (leftLength[1] + x * alive[1]) * (rightLength[0] + (length - 1 - x) * alive[0]);
prob += 1 * totalCommonCases;
if(alive[0] == 1) {
prob += 1 * 1 * totalLength[1];
}
if(alive[1] == 1) {
prob += 1 * totalLength[0] * 1;
}
if(alive[0] == 1 && alive[1] == 1) {
prob -= 2 * 1 * 1 * 1;
}
updateAnswer(current.number + x, prob);
return prob;
};
update(0);
update(length - 1);
{
int low = 0;
int high = length - 1;
while(high - low > 3) {
int mid = (low + high) / 2;
if(update(mid) > update(mid+1)) {
high = mid;
}else {
low = mid;
}
}
for(int x = low; x <= high; x++) {
update(x);
}
}
for(int i= 0; i < 2; i++) {
if(alive[i] > 0) leftLength[i] = leftLength[i] + length;
}
}
}
printf("%d\n", bestAnswer);
}
return 0;
}
#include <vector>
#include <cassert>
#include <utility>
#include <algorithm>
#include <cstdio>
#include <string>
#include <tuple>

struct Interval {
    int l, r;
    int length() const { return r-l+1; }
};

long long pow10[17], ones[17];

int getBestCount(int L, int R, int power, int logPower, int D) {
    if(power == 1) {
        if(L <= D && D <= R) {
            return 1;
        }else {
            return 0;
        }
    }

    if(L == 0) {
        int cnt = 0;
        while(ones[cnt + 1] * D <= R) cnt += 1;
        return cnt; 
    }

    if(R == power - 1) {
        int cnt = logPower, dep = 0;
        while(L <= ones[cnt] * D + ones[dep] * 9 * pow10[cnt]) {
            cnt -= 1;
            dep += 1;
        }
        return cnt;
    }

    const int L0 = L / power, R0 = R / power;
    if(L0 == R0) {
        int childCnt = getBestCount(L % power, R % power, power / 10, logPower - 1, D);
        if(D == L0) childCnt ++;
        return childCnt;
    }
 
    int count = 0;
    for(int digit = L0; digit <= R0; digit++) {
        int childCount;
        if(digit == L0) {
            childCount = getBestCount(L % power, power - 1, power / 10, logPower - 1, D);
        }else if(L0 < digit && digit < R0) {
            childCount = logPower;
        }else {
            childCount = getBestCount(0, R % power, power / 10, logPower - 1, D);
        }
 
        int nextCount = childCount + (digit == D ? 1 : 0);
        if(nextCount < count) continue;
 
        if(nextCount > count) {
            count = nextCount;
        }
    }
 
    return count;
}

std::vector<Interval> genBestIntervals(int L, int R, int power, int logPower, int D) {
    if(power == 1) {
        if(L <= D && D <= R) {
            return std::vector<Interval>(1, Interval{D, D});
        }else {
            return std::vector<Interval>(1, Interval{L, R});
        }
    }
 
    const int L0 = L / power, R0 = R / power;
    if(L0 == R0) {
        auto child = genBestIntervals(L % power, R % power, power / 10, logPower - 1, D);
        if(L0 > 0) for(auto &intv : child) intv.l += L0 * power, intv.r += L0 * power;
        return child;
    }
 
    int count = getBestCount(L, R, power, logPower, D);
    std::vector<Interval> intervals;
    for(int digit = L0; digit <= R0; digit++) {
        int childCount;
        if(digit == L0) {
            childCount = getBestCount(L % power, power - 1, power / 10, logPower - 1, D);
        }else if(L0 < digit && digit < R0) {
            childCount = logPower;
        }else {
            childCount = getBestCount(0, R % power, power / 10, logPower - 1, D);
        }
 
        int nextCount = childCount + (digit == D ? 1 : 0);
        if(nextCount < count) continue;
        assert(nextCount == count);
        
        std::vector<Interval> childIntervals;
        if(digit == L0) {
            childIntervals = genBestIntervals(L % power, power - 1, power / 10, logPower - 1, D);
        }else if(L0 < digit && digit < R0) {
            childIntervals.push_back({(power - 1) / 9 * D, (power - 1) / 9 * D});
        }else {
            childIntervals = genBestIntervals(0, R % power, power / 10, logPower - 1, D);
        }
 
        for(Interval intv : childIntervals) {
            intv.l += digit * power;
            intv.r += digit * power;
            intervals.push_back(intv);
        }
    }
 
    return intervals;
}
 
struct Event {
    int number, who, inserted;
};
 
int main() {
#ifdef IN_MY_COMPUTER
    freopen("example.in.txt", "r", stdin);
#endif
    pow10[0] = 1;
    for(int i = 1; i < 17; i++) pow10[i] = pow10[i-1] * 10, ones[i] = ones[i-1] * 10 + 1;
 
    int T; scanf("%d", &T);
    for(int tc = 1; tc <= T; tc++) {
        int A, B, C, D; scanf("%d%d%d%d", &A,&B,&C,&D);
 
        int logPower = std::to_string(B).length() - 1;
        int power = 1; for(int rep = 0; rep < logPower; rep++) power *= 10;
 
        auto CIntervals = genBestIntervals(A, B, power, logPower, C);
        auto DIntervals = genBestIntervals(A, B, power, logPower, D);
 
        std::vector<Event> events;
        for(auto it : CIntervals) events.push_back({it.l, 0, +1}), events.push_back({it.r+1, 0, -1});
        for(auto it : DIntervals) events.push_back({it.l, 1, +1}), events.push_back({it.r+1, 1, -1});
        std::sort(events.begin(), events.end(), [&](const Event &e1, const Event &e2) { return e1.number < e2.number; });
        events.push_back({B+1, -1, -1});
 
        int bestAnswer = -1;
        long long bestProbability = -1;
 
        auto updateAnswer = [&bestAnswer, &bestProbability](int pos, long long prob) {
            if(prob > bestProbability) {
                bestProbability = prob;
                bestAnswer = pos;
            }else if(prob == bestProbability && bestAnswer > pos) {
                bestAnswer = pos;
            }
        };

 
        long long totalLength[] = {0, 0};
        for(auto &it : CIntervals) totalLength[0] += it.length();
        for(auto &it : DIntervals) totalLength[1] += it.length();
 
        int alive[] = {0, 0};
        long long totalCommonCases = 0;
        for(size_t _idx = 0; _idx+1 < events.size(); _idx++) {
            Event &current = events[_idx];
            Event &next = events[_idx+1];
 
            alive[current.who] = alive[current.who] + current.inserted;
            if(current.number < next.number || next.who < 0) {
                if(alive[0] > 0 && alive[1] > 0) {
                    totalCommonCases += next.number - current.number;
                }
            }
        }

        alive[0] = 0;
        alive[1] = 0;

        long long leftLength[] = {0, 0};
        long long rightLength[] = {totalLength[0], totalLength[1]};
 
        for(size_t _idx = 0; _idx+1 < events.size(); _idx++) {
            Event &current = events[_idx];
            Event &next = events[_idx+1];
 
            alive[current.who] = alive[current.who] + current.inserted;
            if(current.number < next.number || next.who < 0) {
                const long long length = (next.number - current.number);
 
                for(int i= 0; i < 2; i++) {
                    if(alive[i] > 0) rightLength[i] = rightLength[i] - length;
                }
 
                // [5, 8) length=3
                // 5, 6 (7)
                // (5) 6 7
                auto update = [&](int x) {
                    long long prob =
                        3 * (leftLength[0] + x * alive[0]) * (rightLength[1] + (length - 1 - x) * alive[1]) +
                        3 * (leftLength[1] + x * alive[1]) * (rightLength[0] + (length - 1 - x) * alive[0]);
                    prob += 1 * totalCommonCases;
                    if(alive[0] == 1) {
                        prob += 1 * 1 * totalLength[1];
                    }
                    if(alive[1] == 1) {
                        prob += 1 * totalLength[0] * 1;
                    }
                    if(alive[0] == 1 && alive[1] == 1) {
                        prob -= 2 * 1 * 1 * 1;
                    }
                    updateAnswer(current.number + x, prob);
                    return prob;
                };
 
                update(0);
                update(length - 1);
                {
                    int low = 0;
                    int high = length - 1;
                    while(high - low > 3) {
                        int mid = (low + high) / 2;
                        if(update(mid) > update(mid+1)) {
                            high = mid;
                        }else {
                            low = mid;
                        }
                    }
                    for(int x = low; x <= high; x++) {
                        update(x);
                    }
                }
 
                for(int i= 0; i < 2; i++) {
                    if(alive[i] > 0) leftLength[i] = leftLength[i] + length;
                }
            }
        }
 
        printf("%d\n", bestAnswer);
 
    }
 
    return 0;
} 