#include <iostream>
#include <cstdio>
#include <fstream>
#include <string>
#include <cstring>
#include <algorithm>
#include <vector>
#include <stack>
#include <queue>
#include <map>
#include <set>
#include <functional>
#include <numeric>
#include <cstdlib>
#include <iomanip>
#include <cassert>
using namespace std;
const int MAXN = 200005;
const int LOG = 18;
int center, timer, n, m, l, x, y, t, edgesInPath, furthestNode, diameterNode1, diameterNode2;
long long maxDist;
int timeIn[MAXN];
int timeOut[MAXN];
int up[MAXN][19];
bool visited[MAXN];
int edgesFromRoot[MAXN];
long long distanceFromRoot[MAXN];
vector<pair<int, int> > graph[MAXN];
long long componentDiameter, centerFar;
void dfsInit(int v, int parent, int pathEdges, long long pathLength) {
visited[v] = true;
up[v][0] = parent;
distanceFromRoot[v] = pathLength;
edgesFromRoot[v] = pathEdges;
timeIn[v] = ++timer;
for (int i = 1; i <= LOG; i++) {
up[v][i] = up[up[v][i - 1]][i - 1];
}
for (int i = 0; i < (int)graph[v].size(); i++) {
int to = graph[v][i].first;
int len = graph[v][i].second;
if (to != parent) {
dfsInit(to, v, pathEdges + 1, pathLength + len);
}
}
timeOut[v] = ++timer;
}
void dfsDiameter(int v, int parent, int pathEdges, long long pathLength) {
if (pathLength > maxDist) {
maxDist = pathLength;
furthestNode = v;
edgesInPath = pathEdges;
}
for (int i = 0; i < (int)graph[v].size(); i++) {
int to = graph[v][i].first;
int len = graph[v][i].second;
if (to != parent) {
dfsDiameter(to, v, pathEdges + 1, pathLength + len);
}
}
}
struct component {
long long diameter;
int center;
long long centerFar;
component() {}
component(long long _diameter, int _center, long long _centerFar): diameter(_diameter), center(_center), centerFar(_centerFar) {}
};
inline bool operator < (const component & a, const component & b) {
return (a.diameter > b.diameter || (a.diameter == b.diameter && a.center < b.center));
}
vector<component> initialComponents;
multiset<component> forest;
multiset<component>::iterator it;
component componentA, componentB;
bool isAncestor(int x, int y) {
return (timeIn[x] <= timeIn[y] && timeOut[x] >= timeOut[y]);
}
int findLCA(int a, int b) {
if (isAncestor(a, b)) {
return a;
}
if (isAncestor(b, a)) {
return b;
}
for (int i = LOG; i >= 0; i--) {
if (!isAncestor(up[a][i], b)) {
a = up[a][i];
}
}
return up[a][0];
}
long long distanceBetween(int a, int b) {
return distanceFromRoot[a] + distanceFromRoot[b] - 2 * distanceFromRoot[findLCA(a, b)];
}
int edgesBetween(int a, int b) {
return edgesFromRoot[a] + edgesFromRoot[b] - 2 * edgesFromRoot[findLCA(a, b)];
}
int getUp(int v, int jump) {
for (int i = 0; i <= LOG; i++) {
if (jump & (1 << i)) {
v = up[v][i];
}
}
return v;
}
int pathDive(int pathNode1, int pathNode2, int edgesInPath, int dive) {
int LCA = findLCA(pathNode1, pathNode2);
int jump = min(dive, edgesFromRoot[pathNode1] - edgesFromRoot[LCA]);
dive -= jump;
if (dive > 0) {
return getUp(pathNode2, edgesFromRoot[pathNode2] - edgesFromRoot[LCA] - dive);
} else {
return getUp(pathNode1, jump);
}
}
int findCenter(int pathNode1, int pathNode2, int edgesInPath) {
int l, r, mid, iter, pathNode, bestCenter;
long long f1, f2, bestF;
bestF = (long long)9e18;
l = 0; r = edgesInPath;
for (iter = 0; iter < LOG; iter++) {
mid = (l + r) >> 1;
pathNode = pathDive(pathNode1, pathNode2, edgesInPath, mid);
f1 = max(distanceBetween(pathNode, pathNode1), distanceBetween(pathNode, pathNode2));
if (f1 < bestF) {
bestF = f1;
bestCenter = pathNode;
}
if (mid < edgesInPath) {
pathNode = pathDive(pathNode1, pathNode2, edgesInPath, mid + 1);
f2 = max(distanceBetween(pathNode, pathNode1), distanceBetween(pathNode, pathNode2));
if (f2 < bestF) {
bestF = f2;
bestCenter = pathNode;
}
if (f1 < f2) {
r = mid;
} else {
l = mid;
}
}
}
for (int goBack = 1; goBack <= 4; goBack++) {
if (edgesBetween(pathNode1, bestCenter) < goBack) {
break;
}
int attempt = pathDive(bestCenter, pathNode1, edgesBetween(pathNode1, bestCenter), goBack);
f1 = max(distanceBetween(pathNode1, attempt), distanceBetween(pathNode2, attempt));
if (f1 < bestF) {
bestF = f1;
bestCenter = attempt;
}
}
for (int goForward = 1; goForward <= 4; goForward++) {
if (edgesBetween(bestCenter, pathNode2) < goForward) {
break;
}
int attempt = pathDive(bestCenter, pathNode2, edgesBetween(bestCenter, pathNode2), goForward);
f1 = max(distanceBetween(pathNode1, attempt), distanceBetween(pathNode2, attempt));
if (f1 < bestF) {
bestF = f1;
bestCenter = attempt;
}
}
return bestCenter;
}
component mergeComponents(component a, component b) {
int aCenter = a.center;
int bCenter = b.center;
long long aFar = a.centerFar;
long long bFar = b.centerFar;
long long cDiameter = aFar + l + bFar;
int cCenter;
long long cCenterFar;
if (aFar <= bFar) {
cCenter = bCenter;
cCenterFar = max(aFar + l, bFar);
} else {
cCenter = aCenter;
cCenterFar = max(aFar, bFar + l);
}
if (cDiameter < a.diameter) {
cDiameter = a.diameter;
cCenter = a.center;
cCenterFar = a.centerFar;
}
if (cDiameter < b.diameter) {
cDiameter = b.diameter;
cCenter = b.center;
cCenterFar = b.centerFar;
}
return component(cDiameter, cCenter, cCenterFar);
}
int main() {
//freopen("input.txt", "r", stdin);
scanf("%d %d %d", &n, &m, &l);
for (int i = 0; i < m; i++) {
scanf("%d %d %d", &x, &y, &t);
graph[x].push_back(make_pair(y, t));
graph[y].push_back(make_pair(x, t));
}
for (int i = 0; i < n; i++) {
if (visited[i]) {
continue;
}
dfsInit(i, i, 0, 0);
maxDist = furthestNode = -1;
dfsDiameter(i, i, 0, 0);
diameterNode1 = furthestNode;
maxDist = furthestNode = -1;
dfsDiameter(diameterNode1, diameterNode1, 0, 0);
diameterNode2 = furthestNode;
assert(diameterNode1 != -1 && diameterNode2 != -1);
center = findCenter(diameterNode1, diameterNode2, edgesInPath);
componentDiameter = maxDist;
centerFar = max(distanceBetween(diameterNode1, center), distanceBetween(diameterNode2, center));
initialComponents.push_back(component(componentDiameter, center, centerFar));
}
sort(initialComponents.begin(), initialComponents.end());
component ans = initialComponents[0];
for (int i = 1; i < (int)initialComponents.size(); i++) {
ans = mergeComponents(ans, initialComponents[i]);
}
cout << ans.diameter << "\n";
return 0;
}
#include <iostream>
#include <cstdio>
#include <fstream>
#include <string>
#include <cstring>
#include <algorithm>
#include <vector>
#include <stack>
#include <queue>
#include <map>
#include <set>
#include <functional>
#include <numeric>
#include <cstdlib>
#include <iomanip>
#include <cassert>

using namespace std;

const int MAXN = 200005;
const int LOG = 18;

int center, timer, n, m, l, x, y, t, edgesInPath, furthestNode, diameterNode1, diameterNode2;
long long maxDist;
int timeIn[MAXN];
int timeOut[MAXN];
int up[MAXN][19];
bool visited[MAXN];
int edgesFromRoot[MAXN];
long long distanceFromRoot[MAXN];
vector<pair<int, int> > graph[MAXN];
long long componentDiameter, centerFar;

void dfsInit(int v, int parent, int pathEdges, long long pathLength) {
	visited[v] = true;
	up[v][0] = parent;
	distanceFromRoot[v] = pathLength;
	edgesFromRoot[v] = pathEdges;
	timeIn[v] = ++timer;
	for (int i = 1; i <= LOG; i++) {
		up[v][i] = up[up[v][i - 1]][i - 1];
	}
	for (int i = 0; i < (int)graph[v].size(); i++) {
		int to = graph[v][i].first;
		int len = graph[v][i].second;
		if (to != parent) {
			dfsInit(to, v, pathEdges + 1, pathLength + len);
		}
	}
	timeOut[v] = ++timer;
}

void dfsDiameter(int v, int parent, int pathEdges, long long pathLength) {
	if (pathLength > maxDist) {
		maxDist = pathLength;
		furthestNode = v;
		edgesInPath = pathEdges;
	}
	for (int i = 0; i < (int)graph[v].size(); i++) {
		int to = graph[v][i].first;
		int len = graph[v][i].second;
		if (to != parent) {
			dfsDiameter(to, v, pathEdges + 1, pathLength + len);
		}
	}
}

struct component {
	long long diameter;
	int center;
	long long centerFar;
	component() {}
	component(long long _diameter, int _center, long long _centerFar): diameter(_diameter), center(_center), centerFar(_centerFar) {}
};

inline bool operator < (const component & a, const component & b) {
	return (a.diameter > b.diameter || (a.diameter == b.diameter && a.center < b.center));
}

vector<component> initialComponents;
multiset<component> forest;
multiset<component>::iterator it;
component componentA, componentB;

bool isAncestor(int x, int y) {
	return (timeIn[x] <= timeIn[y] && timeOut[x] >= timeOut[y]);
}

int findLCA(int a, int b) {
	if (isAncestor(a, b)) {
		return a;
	}
	if (isAncestor(b, a)) {
		return b;
	}
	for (int i = LOG; i >= 0; i--) {
		if (!isAncestor(up[a][i], b)) {
			a = up[a][i];
		}
	}
	return up[a][0];
}

long long distanceBetween(int a, int b) {
	return distanceFromRoot[a] + distanceFromRoot[b] - 2 * distanceFromRoot[findLCA(a, b)];
}

int edgesBetween(int a, int b) {
	return edgesFromRoot[a] + edgesFromRoot[b] - 2 * edgesFromRoot[findLCA(a, b)];
}

int getUp(int v, int jump) {
	for (int i = 0; i <= LOG; i++) {
		if (jump & (1 << i)) {
			v = up[v][i];
		}
	}
	return v;
}

int pathDive(int pathNode1, int pathNode2, int edgesInPath, int dive) {
	int LCA = findLCA(pathNode1, pathNode2);
	int jump = min(dive, edgesFromRoot[pathNode1] - edgesFromRoot[LCA]);
	dive -= jump;
	if (dive > 0) {
		return getUp(pathNode2, edgesFromRoot[pathNode2] - edgesFromRoot[LCA] - dive);
	} else {
		return getUp(pathNode1, jump);
	}
}

int findCenter(int pathNode1, int pathNode2, int edgesInPath) {
	int l, r, mid, iter, pathNode, bestCenter;
	long long f1, f2, bestF;
	bestF = (long long)9e18;
	l = 0; r = edgesInPath;
	for (iter = 0; iter < LOG; iter++) {
		mid = (l + r) >> 1;
		pathNode = pathDive(pathNode1, pathNode2, edgesInPath, mid);
		f1 = max(distanceBetween(pathNode, pathNode1), distanceBetween(pathNode, pathNode2));
		if (f1 < bestF) {
			bestF = f1;
			bestCenter = pathNode;
		}
		if (mid < edgesInPath) {
			pathNode = pathDive(pathNode1, pathNode2, edgesInPath, mid + 1);
			f2 = max(distanceBetween(pathNode, pathNode1), distanceBetween(pathNode, pathNode2));
			if (f2 < bestF) {
				bestF = f2;
				bestCenter = pathNode;
			}
			if (f1 < f2) {
				r = mid;
			} else {
				l = mid;
			}
		}
	}
	for (int goBack = 1; goBack <= 4; goBack++) {
		if (edgesBetween(pathNode1, bestCenter) < goBack) {
			break;
		}
		int attempt = pathDive(bestCenter, pathNode1, edgesBetween(pathNode1, bestCenter), goBack);
		f1 = max(distanceBetween(pathNode1, attempt), distanceBetween(pathNode2, attempt));
		if (f1 < bestF) {
			bestF = f1;
			bestCenter = attempt;
		}
	}
	for (int goForward = 1; goForward <= 4; goForward++) {
		if (edgesBetween(bestCenter, pathNode2) < goForward) {
			break;
		}
		int attempt = pathDive(bestCenter, pathNode2, edgesBetween(bestCenter, pathNode2), goForward);
		f1 = max(distanceBetween(pathNode1, attempt), distanceBetween(pathNode2, attempt));
		if (f1 < bestF) {
			bestF = f1;
			bestCenter = attempt;
		}
	}
	return bestCenter;
}

component mergeComponents(component a, component b) {
	int aCenter = a.center;
	int bCenter = b.center;
	long long aFar = a.centerFar;
	long long bFar = b.centerFar;
	long long cDiameter = aFar + l + bFar;
	int cCenter;
	long long cCenterFar;
	if (aFar <= bFar) {
		cCenter = bCenter;
		cCenterFar = max(aFar + l, bFar);
	} else {
		cCenter = aCenter;
		cCenterFar = max(aFar, bFar + l);
	}
	if (cDiameter < a.diameter) {
		cDiameter = a.diameter;
		cCenter = a.center;
		cCenterFar = a.centerFar;
	} 
	if (cDiameter < b.diameter) {
		cDiameter = b.diameter;
		cCenter = b.center;
		cCenterFar = b.centerFar;
	}
	return component(cDiameter, cCenter, cCenterFar);
}

int main() {
	//freopen("input.txt", "r", stdin);
	scanf("%d %d %d", &n, &m, &l);
	for (int i = 0; i < m; i++) {
		scanf("%d %d %d", &x, &y, &t);
		graph[x].push_back(make_pair(y, t));
		graph[y].push_back(make_pair(x, t));
	}
	for (int i = 0; i < n; i++) {
		if (visited[i]) {
			continue;
		}
		dfsInit(i, i, 0, 0);
		maxDist = furthestNode = -1;
		dfsDiameter(i, i, 0, 0);
		diameterNode1 = furthestNode;
		maxDist = furthestNode = -1;
		dfsDiameter(diameterNode1, diameterNode1, 0, 0);
		diameterNode2 = furthestNode;
		assert(diameterNode1 != -1 && diameterNode2 != -1);
		center = findCenter(diameterNode1, diameterNode2, edgesInPath);
		componentDiameter = maxDist;
		centerFar = max(distanceBetween(diameterNode1, center), distanceBetween(diameterNode2, center));
		initialComponents.push_back(component(componentDiameter, center, centerFar));
	}
	sort(initialComponents.begin(), initialComponents.end());
	component ans = initialComponents[0];
	for (int i = 1; i < (int)initialComponents.size(); i++) {
		ans = mergeComponents(ans, initialComponents[i]);
	}
	cout << ans.diameter << "\n";
	return 0;
}